axum/README.md

284 lines
7.0 KiB
Markdown
Raw Normal View History

2021-05-31 22:28:26 +08:00
# tower-web
This is *not* https://github.com/carllerche/tower-web even though the name is
the same. Its just a prototype of a minimal HTTP framework I've been toying
2021-06-01 06:34:09 +08:00
with. Will probably change the name to something else.
2021-05-31 22:28:26 +08:00
# What is this?
## Goals
- As easy to use as tide. I don't really consider warp easy to use due to type
tricks it uses. `fn route() -> impl Filter<...>` also isn't very ergonomic.
2021-06-01 18:08:00 +08:00
Just `async fn(Request) -> Response` would be nicer.
2021-05-31 22:28:26 +08:00
- Deep integration with Tower meaning you can
- Apply middleware to the entire application.
- Apply middleware to a single route.
- Apply middleware to subset of routes.
- Just focus on routing and generating responses. Tower can do the rest.
Want timeouts? Use `tower::timeout::Timeout`. Want logging? Use
`tower_http::trace::Trace`.
- Work with Tokio. tide is cool but requires async-std.
- Not macro based. Heavy macro based APIs can be very ergonomic but comes at a
complexity cost. Would like to see if I can design an API that is ergonomic
and doesn't require macros.
## Non-goals
- Runtime independent. If becoming runtime independent isn't too much then fine
but explicitly designing for runtime independence isn't a goal.
- Speed. As long as things are reasonably fast that is fine. For example using
async-trait for ergonomics is fine even though it comes at a cost.
# Example usage
2021-06-01 06:47:12 +08:00
NOTE: Error handling has changed quite a bit and these examples are slightly out
of date. See the examples for working examples.
2021-05-31 22:28:26 +08:00
Defining a single route looks like this:
```rust
let app = tower_web::app().at("/").get(root);
2021-06-01 18:08:00 +08:00
async fn root(req: Request<Body>) -> &'static str {
"Hello, World!"
2021-05-31 22:28:26 +08:00
}
```
Adding more routes follows the same pattern:
```rust
let app = tower_web::app()
.at("/")
.get(root)
.at("/users")
.get(users_index)
.post(users_create);
```
Handler functions are just async functions like:
```rust
2021-06-01 18:08:00 +08:00
async fn handler(req: Request<Body>) -> &'static str {
"Hello, World!"
2021-05-31 22:28:26 +08:00
}
```
2021-06-01 18:08:00 +08:00
They must take the request as the first argument but all arguments following
2021-05-31 22:28:26 +08:00
are called "extractors" and are used to extract data from the request (similar
2021-06-01 18:08:00 +08:00
to rocket but without macros):
2021-05-31 22:28:26 +08:00
```rust
#[derive(Deserialize)]
struct UserPayload {
username: String,
}
#[derive(Deserialize)]
struct Pagination {
page: usize,
per_page: usize,
}
async fn handler(
req: Request<Body>,
// deserialize response body with `serde_json` into a `UserPayload`
user: extract::Json<UserPayload>,
// deserialize query string into a `Pagination`
pagination: extract::Query<Pagination>,
2021-06-01 18:08:00 +08:00
) -> &'static str {
2021-06-01 21:12:22 +08:00
let user: UserPayload = user.0;
let pagination: Pagination = pagination.0;
2021-05-31 22:28:26 +08:00
// ...
}
```
2021-05-31 22:32:56 +08:00
The inputs can also be optional:
```rust
async fn handler(
req: Request<Body>,
user: Option<extract::Json<UserPayload>>,
2021-06-01 18:08:00 +08:00
) -> &'static str {
2021-05-31 22:32:56 +08:00
// ...
}
```
2021-05-31 22:28:26 +08:00
You can also get the raw response body:
```rust
async fn handler(
req: Request<Body>,
// buffer the whole request body
body: Bytes,
2021-06-01 18:08:00 +08:00
) -> &'static str {
2021-05-31 22:28:26 +08:00
// ...
}
```
Or limit the body size:
```rust
async fn handler(
req: Request<Body>,
// max body size in bytes
body: extract::BytesMaxLength<1024>,
2021-06-01 18:08:00 +08:00
) -> &'static str {
// ...
}
```
Params from dynamic routes like `GET /users/:id` can be extracted like so
```rust
async fn handle(
req: Request<Body>,
// get a map of key value pairs
map: extract::UrlParamsMap,
) -> &'static str {
let raw_id: Option<&str> = map.get("id");
let parsed_id: Option<i32> = map.get_typed::<i32>("id");
// ...
}
async fn handle(
req: Request<Body>,
// or get a tuple with each param
params: extract::UrlParams<(i32, String)>,
) -> &'static str {
2021-06-01 21:12:22 +08:00
let (id, name) = params.0;
2021-06-01 18:08:00 +08:00
// ...
2021-05-31 22:28:26 +08:00
}
```
2021-06-01 21:12:22 +08:00
If you wanna go all out you can even deconstruct the extractor directly in the
function signature:
```rust
async fn handle(
req: Request<Body>,
UrlParams((id, name)): UrlParams<(i32, String)>,
) -> &'static str {
// ...
}
```
2021-05-31 22:28:26 +08:00
Anything that implements `FromRequest` can work as an extractor where
2021-06-01 18:08:00 +08:00
`FromRequest` is an async trait:
2021-05-31 22:28:26 +08:00
```rust
#[async_trait]
pub trait FromRequest: Sized {
2021-06-01 18:08:00 +08:00
type Rejection: IntoResponse<B>;
async fn from_request(req: &mut Request<Body>) -> Result<Self, Self::Rejection>;
2021-05-31 22:28:26 +08:00
}
```
2021-05-31 22:32:56 +08:00
This "extractor" pattern is inspired by Bevy's ECS. The idea is that it should
2021-06-01 18:08:00 +08:00
be easy to pick apart the request without having to repeat yourself a lot or use
macros.
2021-05-31 22:32:56 +08:00
2021-06-01 18:08:00 +08:00
The return type must implement `IntoResponse`:
2021-05-31 22:28:26 +08:00
```rust
2021-06-01 18:08:00 +08:00
async fn empty_response(req: Request<Body>) {
// ...
}
// gets `content-type: text/plain`
async fn string_response(req: Request<Body>) -> String {
2021-05-31 22:28:26 +08:00
// ...
}
// gets `content-type: appliation/json`. `Json` can contain any `T: Serialize`
2021-06-01 18:08:00 +08:00
async fn json_response(req: Request<Body>) -> response::Json<User> {
2021-05-31 22:28:26 +08:00
// ...
}
// gets `content-type: text/html`. `Html` can contain any `T: Into<Bytes>`
2021-06-01 18:08:00 +08:00
async fn html_response(req: Request<Body>) -> response::Html<String> {
2021-05-31 22:28:26 +08:00
// ...
}
// or for full control
2021-06-01 18:08:00 +08:00
async fn response(req: Request<Body>) -> Response<Body> {
// ...
}
// Result is supported if each type implements `IntoResponse`
async fn response(req: Request<Body>) -> Result<Html<String>, StatusCode> {
2021-05-31 22:28:26 +08:00
// ...
}
```
2021-06-01 18:08:00 +08:00
This makes error handling quite simple. Basically handlers are not allowed to
fail and must always produce a response. This also means users are in charge of
how their errors are mapped to responses rather than a framework providing some
opaque catch all error type.
2021-05-31 22:28:26 +08:00
You can also apply Tower middleware to single routes:
```rust
let app = tower_web::app()
.at("/")
2021-06-01 18:08:00 +08:00
.get(send_some_large_file.layer(CompressionLayer::new()))
2021-05-31 22:28:26 +08:00
```
Or to the whole app:
```rust
let service = tower_web::app()
.at("/")
.get(root)
.into_service()
let app = ServiceBuilder::new()
.timeout(Duration::from_secs(30))
.layer(TraceLayer::new_for_http())
.layer(CompressionLayer::new())
.service(app);
```
And of course run it with Hyper:
```rust
#[tokio::main]
async fn main() {
tracing_subscriber::fmt::init();
// build our application with some routes
let app = tower_web::app()
.at("/")
.get(handler)
// convert it into a `Service`
.into_service();
// add some middleware
let app = ServiceBuilder::new()
.layer(TraceLayer::new_for_http())
.service(app);
// run it
let addr = SocketAddr::from(([127, 0, 0, 1], 3000));
tracing::debug!("listening on {}", addr);
let server = Server::bind(&addr).serve(Shared::new(app));
server.await.unwrap();
}
```
See the examples directory for more examples.
# TODO
- `RouteBuilder` should have an `async fn serve(self) -> Result<(),
hyper::Error>` for users who just wanna create a hyper server and not care
about the lower level details. Should be gated by a `hyper` feature.
- Each new route makes a new allocation for the response body, since `Or` needs
to unify the response body types. Would be nice to find a way to avoid that.
- It should be possible to package some routes together and apply a tower
middleware to that collection and then merge those routes into the app.