Routing
Supports get
post
delete
patch
put
options
head
connect
trace
.
Supports any
for above APIs.
Supports scope
for scope routes.
Supports resource
and resources
for resourceful routes.
Supports middleware
WIP
```rust extern crate trek_router;
use trek_router::Router;
type F = fn() -> usize;
let mut router = Router::
// scope v1 router.scope("/v1", |v1| { v1.get("/login", || 0); v1.post("/submit", || 1); v1.delete("/read", || 2); });
// scope v2 router.scope("/v2", |v2| { v2.get("/login", || 0); v2.post("/submit", || 1); v2.delete("/read", || 2); });
router.get("/foo", || 3); router.post("/bar", || 4); router.delete("/baz", || 5);
dbg!(&router); ```
```rust extern crate futures; extern crate hyper; extern crate trek_router;
use futures::Future; use hyper::server::Server; use hyper::service::servicefnok; use hyper::{Body, Request, Response, StatusCode}; use std::sync::Arc; use trek_router::Router;
type Params<'a> = Vec<(&'a str, &'a str)>;
type Handler = fn(Request
, Params) -> Body;fn v1login(: Request
, _: Params) -> Body { Body::from("v1 login") }fn v1submit(req: Request
, _: Params) -> Body { Body::from("v1 submit") }fn v1read(req: Request
, _: Params) -> Body { Body::from("v1 read") }fn v2login(: Request
, _: Params) -> Body { Body::from("v2 login") }fn v2submit(req: Request
, _: Params) -> Body { Body::from("v2 submit") }fn v2read(req: Request
, _: Params) -> Body { Body::from("v2 read") }fn users(_req: Request
, _: Params) -> Body { Body::from("users") }fn foo(_: Request
, _: Params) -> Body { Body::from("foo") }fn bar(_req: Request
, _: Params) -> Body { Body::from("bar") }fn baz(_req: Request
, _: Params) -> Body { Body::from("baz") }fn main() { let addr = ([127, 0, 0, 1], 3000).into();
let mut router = Router::<Handler>::new();
router
// scope v1
.scope("/v1", |v1| {
v1.get("/login", v1_login)
.post("/submit", v1_submit)
.delete("/read", v1_read);
})
// scope v2
.scope("/v2", |v2| {
v2.get("/login", v2_login)
.post("/submit", v2_submit)
.delete("/read", v2_read)
// scope users
.scope("users", |u| {
u.any("", users);
});
})
.get("/foo", foo)
.post("/bar", bar)
.delete("/baz", baz);
let router = Arc::new(router);
let routing = move || {
let router = Arc::clone(&router);
service_fn_ok(move |req| {
let method = req.method().to_owned();
let path = req.uri().path().to_owned();
match router.find(&method, &path) {
Some((handler, params)) => Response::new(handler(req, params)),
None => Response::builder()
.status(StatusCode::NOT_FOUND)
.body(Body::from("Not Found"))
.unwrap(),
}
})
};
let server = Server::bind(&addr)
.serve(routing)
.map_err(|e| eprintln!("server error: {}", e));
hyper::rt::run(server);
} ```
| HTTP Verb | Path | Action | | --------- | -------------------- | ------- | | GET | /geocoder/new | new | | POST | /geocoder | create | | GET | /geocoder | show | | GET | /geocoder/edit | edit | | PATCH/PUT | /geocoder | update | | DELETE | /geocoder | destroy | | GET | /users | index | | GET | /users/new | new | | POST | /users | create | | GET | /users/:userid | show | | GET | /users/:userid/edit | edit | | PATCH/PUT | /users/:userid | update | | DELETE | /users/:userid | destroy |
```rust extern crate futures; extern crate hyper; extern crate trek_router;
use futures::Future; use hyper::server::Server; use hyper::service::servicefnok; use hyper::{Body, Request, Response, StatusCode}; use std::sync::Arc; use trek_router::{Resource, ResourceOptions, Resources, Router};
type Params = Vec<(String, String)>; type Handler = fn(Context) -> Body;
struct Context { request: Request
, params: Params, }struct Geocoder {}
impl Resource for Geocoder { type Context = Context; type Body = Body;
fn show(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("Geocoder Show!");
Body::from(s)
}
fn create(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("Geocoder Create!");
Body::from(s)
}
fn update(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("Geocoder Update!");
Body::from(s)
}
fn delete(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("Geocoder Delete!");
Body::from(s)
}
fn edit(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("Geocoder Edit!");
Body::from(s)
}
fn new(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("Geocoder New!");
Body::from(s)
}
}
struct Users {}
impl Resources for Users { type Context = Context; type Body = Body;
fn index(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("Users Index!");
Body::from(s)
}
fn create(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("User Create!");
Body::from(s)
}
fn new(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("User New!");
Body::from(s)
}
fn show(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("User Show, ");
for (k, v) in ctx.params {
s.push_str(&format!("{} = {}", k, v));
}
s.push_str("!");
Body::from(s)
}
fn update(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("User Update, ");
for (k, v) in ctx.params {
s.push_str(&format!("{} = {}", k, v));
}
s.push_str("!");
Body::from(s)
}
fn delete(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("User Delete, ");
for (k, v) in ctx.params {
s.push_str(&format!("{} = {}", k, v));
}
s.push_str("!");
Body::from(s)
}
fn edit(ctx: Self::Context) -> Self::Body {
let mut s = String::new();
s.push_str(&ctx.request.uri().path().to_owned());
s.push_str("\n");
s.push_str("User Edit, ");
for (k, v) in ctx.params {
s.push_str(&format!("{} = {}", k, v));
}
s.push_str("!");
Body::from(s)
}
}
fn main() { let addr = ([127, 0, 0, 1], 3000).into();
let mut router = Router::<Handler>::new();
router.resource("/geocoder", Geocoder::build(ResourceOptions::default()));
router.resources("/users", Users::build(ResourceOptions::default()));
let router = Arc::new(router);
let routing = move || {
let router = Arc::clone(&router);
service_fn_ok(move |request| {
let method = request.method().to_owned();
let path = request.uri().path().to_owned();
match router.find(&method, &path) {
Some((handler, params)) => Response::new(handler(Context {
request,
params: params
.iter()
.map(|(a, b)| (a.to_string(), b.to_string()))
.collect(),
})),
None => Response::builder()
.status(StatusCode::NOT_FOUND)
.body(Body::from("Not Found"))
.unwrap(),
}
})
};
let server = Server::bind(&addr)
.serve(routing)
.map_err(|e| eprintln!("server error: {}", e));
hyper::rt::run(server);
} ```
This project is licensed under either of