1
0
mirror of https://github.com/actix/actix-extras.git synced 2025-01-25 00:01:48 +01:00

323 lines
9.8 KiB
Markdown
Raw Normal View History

# Request & Response
2017-12-02 11:41:20 -08:00
2017-12-04 20:38:38 -08:00
## Response
2018-03-28 22:16:01 +02:00
A builder-like pattern is used to construct an instance of `HttpResponse`.
`HttpResponse` provides several methods that return a `HttpResponseBuilder` instance,
which implements various convenience methods for building responses.
> Check the [documentation](../actix_web/dev/struct.HttpResponseBuilder.html)
> for type descriptions.
The methods `.body`, `.finish`, and `.json` finalize response creation and
return a constructed *HttpResponse* instance. If this methods is called on the same
2018-03-28 22:16:01 +02:00
builder instance multiple times, the builder will panic.
2017-12-04 20:38:38 -08:00
```rust
# extern crate actix_web;
2018-03-30 17:31:18 -07:00
use actix_web::{HttpRequest, HttpResponse, http::ContentEncoding};
2017-12-04 20:38:38 -08:00
fn index(req: HttpRequest) -> HttpResponse {
HttpResponse::Ok()
.content_encoding(ContentEncoding::Br)
.content_type("plain/text")
.header("X-Hdr", "sample")
.body("data")
2017-12-04 20:38:38 -08:00
}
# fn main() {}
```
2017-12-02 11:41:20 -08:00
## Content encoding
Actix automatically *compresses*/*decompresses* payloads. The following codecs are supported:
2017-12-02 11:41:20 -08:00
* Brotli
* Gzip
* Deflate
* Identity
2018-03-28 22:16:01 +02:00
If request headers contain a `Content-Encoding` header, the request payload is decompressed
according to the header value. Multiple codecs are not supported, i.e: `Content-Encoding: br, gzip`.
2018-03-28 22:16:01 +02:00
Response payload is compressed based on the *content_encoding* parameter.
By default, `ContentEncoding::Auto` is used. If `ContentEncoding::Auto` is selected,
then the compression depends on the request's `Accept-Encoding` header.
> `ContentEncoding::Identity` can be used to disable compression.
> If another content encoding is selected, the compression is enforced for that codec.
For example, to enable `brotli` use `ContentEncoding::Br`:
2017-12-03 20:09:46 -08:00
```rust
2017-12-04 16:26:40 -08:00
# extern crate actix_web;
2018-03-30 17:31:18 -07:00
use actix_web::{HttpRequest, HttpResponse, http::ContentEncoding};
2017-12-03 18:58:15 -08:00
fn index(req: HttpRequest) -> HttpResponse {
HttpResponse::Ok()
.content_encoding(ContentEncoding::Br)
.body("data")
2017-12-03 18:58:15 -08:00
}
2017-12-03 20:09:46 -08:00
# fn main() {}
2017-12-03 18:58:15 -08:00
```
2017-12-20 15:12:43 -08:00
## JSON Request
2018-03-31 09:18:25 -07:00
There are several options for json body deserialization.
2017-12-20 15:12:43 -08:00
The first option is to use *Json* extractor. First, you define a handler function
that accepts `Json<T>` as a parameter, then, you use the `.with()` method for registering
2018-03-31 09:18:25 -07:00
this handler. It is also possible to accept arbitrary valid json object by
using `serde_json::Value` as a type `T`.
2018-03-31 09:18:25 -07:00
```rust
# extern crate actix_web;
#[macro_use] extern crate serde_derive;
use actix_web::{App, Json, Result, http};
#[derive(Deserialize)]
struct Info {
username: String,
}
/// extract `Info` using serde
fn index(info: Json<Info>) -> Result<String> {
Ok(format!("Welcome {}!", info.username))
}
fn main() {
let app = App::new().resource(
"/index.html",
|r| r.method(http::Method::POST).with(index)); // <- use `with` extractor
}
```
Another option is to use *HttpResponse::json()*. This method returns a
2018-03-28 22:16:01 +02:00
[*JsonBody*](../actix_web/dev/struct.JsonBody.html) object which resolves into
the deserialized value.
2017-12-20 15:12:43 -08:00
2017-12-20 20:30:54 -08:00
```rust
# extern crate actix;
# extern crate actix_web;
# extern crate futures;
# extern crate serde_json;
# #[macro_use] extern crate serde_derive;
# use actix_web::*;
# use futures::Future;
#[derive(Debug, Serialize, Deserialize)]
2017-12-20 15:12:43 -08:00
struct MyObj {
name: String,
number: i32,
}
2017-12-20 20:30:54 -08:00
fn index(mut req: HttpRequest) -> Box<Future<Item=HttpResponse, Error=Error>> {
req.json().from_err()
.and_then(|val: MyObj| {
println!("model: {:?}", val);
Ok(HttpResponse::Ok().json(val)) // <- send response
2017-12-20 20:30:54 -08:00
})
.responder()
}
# fn main() {}
2017-12-20 15:12:43 -08:00
```
You may also manually load the payload into memory and then deserialize it.
In the following example, we will deserialize a *MyObj* struct. We need to load the request
2018-03-28 22:16:01 +02:00
body first and then deserialize the json into an object.
2017-12-20 15:12:43 -08:00
2017-12-25 08:12:13 -08:00
```rust
# extern crate actix_web;
# extern crate futures;
# use actix_web::*;
# #[macro_use] extern crate serde_derive;
extern crate serde_json;
use futures::{Future, Stream};
#[derive(Serialize, Deserialize)]
struct MyObj {name: String, number: i32}
2018-02-26 08:02:58 -08:00
fn index(req: HttpRequest) -> Box<Future<Item=HttpResponse, Error=Error>> {
2017-12-20 16:05:07 -08:00
// `concat2` will asynchronously read each chunk of the request body and
// return a single, concatenated, chunk
2018-02-26 08:02:58 -08:00
req.concat2()
2017-12-20 16:05:07 -08:00
// `Future::from_err` acts like `?` in that it coerces the error type from
// the future into the final error type
.from_err()
// `Future::and_then` can be used to merge an asynchronous workflow with a
// synchronous workflow
2017-12-20 16:13:21 -08:00
.and_then(|body| { // <- body is loaded, now we can deserialize json
let obj = serde_json::from_slice::<MyObj>(&body)?;
Ok(HttpResponse::Ok().json(obj)) // <- send response
2017-12-20 16:05:07 -08:00
})
2017-12-25 08:12:13 -08:00
.responder()
2017-12-20 15:12:43 -08:00
}
2017-12-25 08:12:13 -08:00
# fn main() {}
2017-12-20 15:12:43 -08:00
```
> A complete example for both options is available in
> [examples directory](https://github.com/actix/actix-web/tree/master/examples/json/).
2017-12-20 15:12:43 -08:00
2017-12-03 18:51:52 -08:00
## JSON Response
2018-03-31 09:18:25 -07:00
The `Json` type allows to respond with well-formed JSON data: simply return a value of
type Json<T> where `T` is the type of a structure to serialize into *JSON*.
The type `T` must implement the `Serialize` trait from *serde*.
2017-12-03 18:51:52 -08:00
```rust
2017-12-04 16:26:40 -08:00
# extern crate actix_web;
2017-12-03 18:51:52 -08:00
#[macro_use] extern crate serde_derive;
2018-03-31 00:16:55 -07:00
use actix_web::{App, HttpRequest, Json, Result, http::Method};
2017-12-03 18:51:52 -08:00
#[derive(Serialize)]
struct MyObj {
name: String,
}
fn index(req: HttpRequest) -> Result<Json<MyObj>> {
Ok(Json(MyObj{name: req.match_info().query("name")?}))
}
fn main() {
2018-03-31 00:16:55 -07:00
App::new()
.resource(r"/a/{name}", |r| r.method(Method::GET).f(index))
2017-12-03 18:51:52 -08:00
.finish();
}
```
2017-12-13 22:36:28 -08:00
## Chunked transfer encoding
2018-03-28 22:16:01 +02:00
Actix automatically decodes *chunked* encoding. `HttpRequest::payload()` already contains
the decoded byte stream. If the request payload is compressed with one of the supported
compression codecs (br, gzip, deflate), then the byte stream is decompressed.
2017-12-13 22:36:28 -08:00
Chunked encoding on a response can be enabled with `HttpResponseBuilder::chunked()`.
This takes effect only for `Body::Streaming(BodyStream)` or `Body::StreamingContext` bodies.
If the response payload compression is enabled and a streaming body is used, chunked encoding
2018-03-28 22:16:01 +02:00
is enabled automatically.
2017-12-13 22:36:28 -08:00
> Enabling chunked encoding for *HTTP/2.0* responses is forbidden.
2017-12-13 22:36:28 -08:00
```rust
2018-02-26 16:41:57 -08:00
# extern crate bytes;
2017-12-13 22:36:28 -08:00
# extern crate actix_web;
2018-02-26 08:02:58 -08:00
# extern crate futures;
# use futures::Stream;
2017-12-13 22:36:28 -08:00
use actix_web::*;
2018-02-26 16:41:57 -08:00
use bytes::Bytes;
use futures::stream::once;
2017-12-13 22:36:28 -08:00
fn index(req: HttpRequest) -> HttpResponse {
HttpResponse::Ok()
.chunked()
.body(Body::Streaming(Box::new(once(Ok(Bytes::from_static(b"data"))))))
2017-12-13 22:36:28 -08:00
}
# fn main() {}
```
## Multipart body
2018-03-28 22:16:01 +02:00
Actix provides multipart stream support.
[*Multipart*](../actix_web/multipart/struct.Multipart.html) is implemented as
a stream of multipart items. Each item can be a
2018-03-28 22:16:01 +02:00
[*Field*](../actix_web/multipart/struct.Field.html) or a nested *Multipart* stream.
`HttpResponse::multipart()` returns the *Multipart* stream for the current request.
2017-12-19 09:55:49 -08:00
The following demonstrates multipart stream handling for a simple form:
2017-12-19 09:55:49 -08:00
2017-12-19 10:10:03 -08:00
```rust,ignore
# extern crate actix_web;
use actix_web::*;
fn index(req: HttpRequest) -> Box<Future<...>> {
req.multipart() // <- get multipart stream for current request
.and_then(|item| { // <- iterate over multipart items
match item {
// Handle multipart Field
multipart::MultipartItem::Field(field) => {
2018-01-13 11:17:48 -08:00
println!("==== FIELD ==== {:?} {:?}", field.headers(), field.content_type());
2018-03-28 22:16:01 +02:00
2017-12-19 10:10:03 -08:00
Either::A(
// Field in turn is a stream of *Bytes* objects
field.map(|chunk| {
println!("-- CHUNK: \n{}",
std::str::from_utf8(&chunk).unwrap());})
.fold((), |_, _| result(Ok(()))))
},
multipart::MultipartItem::Nested(mp) => {
2018-03-28 22:16:01 +02:00
// Or item could be nested Multipart stream
2017-12-19 10:10:03 -08:00
Either::B(result(Ok(())))
}
}
})
}
```
> A full example is available in the
> [examples directory](https://github.com/actix/actix-web/tree/master/examples/multipart/).
2017-12-13 22:36:28 -08:00
## Urlencoded body
2018-03-28 22:16:01 +02:00
Actix provides support for *application/x-www-form-urlencoded* encoded bodies.
`HttpResponse::urlencoded()` returns a
[*UrlEncoded*](../actix_web/dev/struct.UrlEncoded.html) future, which resolves
to the deserialized instance. The type of the instance must implement the
`Deserialize` trait from *serde*.
The *UrlEncoded* future can resolve into an error in several cases:
2017-12-19 14:03:01 -08:00
* content type is not `application/x-www-form-urlencoded`
* transfer encoding is `chunked`.
* content-length is greater than 256k
* payload terminates with error.
```rust
# extern crate actix_web;
# extern crate futures;
2018-04-02 15:08:49 -07:00
#[macro_use] extern crate serde_derive;
2017-12-19 14:03:01 -08:00
use actix_web::*;
use futures::future::{Future, ok};
2018-04-02 15:08:49 -07:00
#[derive(Deserialize)]
struct FormData {
username: String,
}
2017-12-19 14:03:01 -08:00
fn index(mut req: HttpRequest) -> Box<Future<Item=HttpResponse, Error=Error>> {
2018-04-02 15:08:49 -07:00
req.urlencoded::<FormData>() // <- get UrlEncoded future
2017-12-20 21:06:04 -08:00
.from_err()
2018-04-02 15:08:49 -07:00
.and_then(|data| { // <- deserialized instance
println!("USERNAME: {:?}", data.username);
ok(HttpResponse::Ok().into())
2017-12-20 21:06:04 -08:00
})
.responder()
2017-12-19 14:03:01 -08:00
}
# fn main() {}
```
2017-12-13 22:36:28 -08:00
## Streaming request
2018-03-28 22:16:01 +02:00
*HttpRequest* is a stream of `Bytes` objects. It can be used to read the request
body payload.
2017-12-19 00:18:57 -08:00
In the following example, we read and print the request payload chunk by chunk:
2017-12-19 00:18:57 -08:00
```rust
# extern crate actix_web;
# extern crate futures;
# use futures::future::result;
use actix_web::*;
use futures::{Future, Stream};
fn index(mut req: HttpRequest) -> Box<Future<Item=HttpResponse, Error=Error>> {
req.from_err()
2017-12-20 21:06:04 -08:00
.fold((), |_, chunk| {
println!("Chunk: {:?}", chunk);
result::<_, error::PayloadError>(Ok(()))
})
.map(|_| HttpResponse::Ok().finish())
2017-12-20 21:06:04 -08:00
.responder()
2017-12-19 00:18:57 -08:00
}
# fn main() {}
```