rama_http/layer/decompression/
mod.rs

1//! Middleware that decompresses request and response bodies.
2//!
3//! # Examples
4//!
5//! #### Request
6//! ```rust
7//! use std::{error::Error, io::Write};
8//!
9//! use bytes::{Bytes, BytesMut};
10//! use flate2::{write::GzEncoder, Compression};
11//!
12//! use rama_http::{Body, header, HeaderValue, Request, Response};
13//! use rama_core::service::service_fn;
14//! use rama_core::{Context, Service, Layer};
15//! use rama_http::layer::decompression::{DecompressionBody, RequestDecompressionLayer};
16//! use rama_http::dep::http_body_util::BodyExt;
17//! use rama_core::error::BoxError;
18//!
19//! # #[tokio::main]
20//! # async fn main() -> Result<(), BoxError> {
21//! // A request encoded with gzip coming from some HTTP client.
22//! let mut encoder = GzEncoder::new(Vec::new(), Compression::default());
23//! encoder.write_all(b"Hello?")?;
24//! let request = Request::builder()
25//!     .header(header::CONTENT_ENCODING, "gzip")
26//!     .body(Body::from(encoder.finish()?))?;
27//!
28//! // Our HTTP server
29//! let mut server = (
30//!     // Automatically decompress request bodies.
31//!     RequestDecompressionLayer::new(),
32//! ).into_layer(service_fn(handler));
33//!
34//! // Send the request, with the gzip encoded body, to our server.
35//! let _response = server.serve(Context::default(), request).await?;
36//!
37//! // Handler receives request whose body is decoded when read
38//! async fn handler(mut req: Request<DecompressionBody<Body>>) -> Result<Response, BoxError>{
39//!     let data = req.into_body().collect().await?.to_bytes();
40//!     assert_eq!(&data[..], b"Hello?");
41//!     Ok(Response::new(Body::from("Hello, World!")))
42//! }
43//! # Ok(())
44//! # }
45//! ```
46//!
47//! #### Response
48//!
49//! ```rust
50//! use std::convert::Infallible;
51//!
52//! use bytes::{Bytes, BytesMut};
53//!
54//! use rama_http::{Body, Request, Response};
55//! use rama_core::service::service_fn;
56//! use rama_core::{Context, Service, Layer};
57//! use rama_http::layer::{compression::Compression, decompression::DecompressionLayer};
58//! use rama_http::dep::http_body_util::BodyExt;
59//! use rama_core::error::BoxError;
60//!
61//! #
62//! # #[tokio::main]
63//! # async fn main() -> Result<(), BoxError> {
64//! # async fn handle(req: Request) -> Result<Response, Infallible> {
65//! #     let body = Body::from("Hello, World!");
66//! #     Ok(Response::new(body))
67//! # }
68//!
69//! // Some opaque service that applies compression.
70//! let service = Compression::new(service_fn(handle));
71//!
72//! // Our HTTP client.
73//! let mut client = (
74//!     // Automatically decompress response bodies.
75//!     DecompressionLayer::new(),
76//! ).into_layer(service);
77//!
78//! // Call the service.
79//! //
80//! // `DecompressionLayer` takes care of setting `Accept-Encoding`.
81//! let request = Request::new(Body::default());
82//!
83//! let response = client
84//!     .serve(Context::default(), request)
85//!     .await?;
86//!
87//! // Read the body
88//! let body = response.into_body();
89//! let bytes = body.collect().await?.to_bytes().to_vec();
90//! let body = String::from_utf8(bytes).map_err(Into::<BoxError>::into)?;
91//!
92//! assert_eq!(body, "Hello, World!");
93//! #
94//! # Ok(())
95//! # }
96//! ```
97
98mod request;
99
100pub(crate) mod body;
101mod layer;
102mod service;
103
104#[doc(inline)]
105pub use self::{body::DecompressionBody, layer::DecompressionLayer, service::Decompression};
106
107#[doc(inline)]
108pub use self::request::layer::RequestDecompressionLayer;
109#[doc(inline)]
110pub use self::request::service::RequestDecompression;
111
112#[cfg(test)]
113mod tests {
114    use super::*;
115
116    use std::convert::Infallible;
117    use std::io::Write;
118
119    use crate::dep::http_body_util::BodyExt;
120    use crate::layer::compression::Compression;
121    use crate::{Body, HeaderMap, HeaderName, Request, Response};
122    use rama_core::service::service_fn;
123    use rama_core::{Context, Service};
124
125    use flate2::write::GzEncoder;
126
127    #[tokio::test]
128    async fn works() {
129        let client = Decompression::new(Compression::new(service_fn(handle)));
130
131        let req = Request::builder()
132            .header("accept-encoding", "gzip")
133            .body(Body::empty())
134            .unwrap();
135        let res = client.serve(Context::default(), req).await.unwrap();
136
137        // read the body, it will be decompressed automatically
138        let body = res.into_body();
139        let collected = body.collect().await.unwrap();
140        let decompressed_data = String::from_utf8(collected.to_bytes().to_vec()).unwrap();
141
142        assert_eq!(decompressed_data, "Hello, World!");
143    }
144
145    async fn handle(_req: Request) -> Result<Response, Infallible> {
146        let mut trailers = HeaderMap::new();
147        trailers.insert(HeaderName::from_static("foo"), "bar".parse().unwrap());
148        let body = Body::from("Hello, World!");
149        Ok(Response::builder().body(body).unwrap())
150    }
151
152    #[tokio::test]
153    async fn decompress_multi_gz() {
154        let client = Decompression::new(service_fn(handle_multi_gz));
155
156        let req = Request::builder()
157            .header("accept-encoding", "gzip")
158            .body(Body::empty())
159            .unwrap();
160        let res = client.serve(Context::default(), req).await.unwrap();
161
162        // read the body, it will be decompressed automatically
163        let body = res.into_body();
164        let decompressed_data =
165            String::from_utf8(body.collect().await.unwrap().to_bytes().to_vec()).unwrap();
166
167        assert_eq!(decompressed_data, "Hello, World!");
168    }
169
170    #[tokio::test]
171    async fn decompress_multi_zstd() {
172        let client = Decompression::new(service_fn(handle_multi_zstd));
173
174        let req = Request::builder()
175            .header("accept-encoding", "zstd")
176            .body(Body::empty())
177            .unwrap();
178        let res = client.serve(Context::default(), req).await.unwrap();
179
180        // read the body, it will be decompressed automatically
181        let body = res.into_body();
182        let decompressed_data =
183            String::from_utf8(body.collect().await.unwrap().to_bytes().to_vec()).unwrap();
184
185        assert_eq!(decompressed_data, "Hello, World!");
186    }
187
188    async fn handle_multi_gz(_req: Request<Body>) -> Result<Response<Body>, Infallible> {
189        let mut buf = Vec::new();
190        let mut enc1 = GzEncoder::new(&mut buf, Default::default());
191        enc1.write_all(b"Hello, ").unwrap();
192        enc1.finish().unwrap();
193
194        let mut enc2 = GzEncoder::new(&mut buf, Default::default());
195        enc2.write_all(b"World!").unwrap();
196        enc2.finish().unwrap();
197
198        let mut res = Response::new(Body::from(buf));
199        res.headers_mut()
200            .insert("content-encoding", "gzip".parse().unwrap());
201        Ok(res)
202    }
203
204    async fn handle_multi_zstd(_req: Request<Body>) -> Result<Response<Body>, Infallible> {
205        let mut buf = Vec::new();
206        let mut enc1 = zstd::Encoder::new(&mut buf, Default::default()).unwrap();
207        enc1.write_all(b"Hello, ").unwrap();
208        enc1.finish().unwrap();
209
210        let mut enc2 = zstd::Encoder::new(&mut buf, Default::default()).unwrap();
211        enc2.write_all(b"World!").unwrap();
212        enc2.finish().unwrap();
213
214        let mut res = Response::new(Body::from(buf));
215        res.headers_mut()
216            .insert("content-encoding", "zstd".parse().unwrap());
217        Ok(res)
218    }
219}