make body return borrowed decoder

This commit is contained in:
Ashley Mannix
2017-08-19 16:32:00 +10:00
parent e2fa97254e
commit 2cb70c872a
7 changed files with 69 additions and 23 deletions

View File

@@ -1,17 +1,40 @@
#![deny(warnings)]
#![allow(warnings)] // remove when error_chain is fixed
extern crate futures;
extern crate reqwest;
extern crate tokio_core;
#[macro_use]
extern crate error_chain;
use futures::Future;
use std::mem;
use std::io::{self, Cursor};
use futures::{Future, Stream};
use reqwest::unstable::async::{Client, Decoder};
error_chain! {
foreign_links {
ReqError(reqwest::Error);
IoError(io::Error);
}
}
fn main() {
let mut core = tokio_core::reactor::Core::new().unwrap();
let client = reqwest::unstable::async::Client::new(&core.handle()).unwrap();
let client = Client::new(&core.handle()).unwrap();
let work = client.get("https://hyper.rs").unwrap().send().map(|res| {
println!("{}", res.status());
});
let work = client.get("https://hyper.rs").unwrap()
.send()
.map_err(|e| Error::from(e))
.and_then(|mut res| {
println!("{}", res.status());
let body = mem::replace(res.body_mut(), Decoder::empty());
body.concat2().map_err(Into::into)
})
.and_then(|body| {
let mut body = Cursor::new(body);
io::copy(&mut body, &mut io::stdout()).map_err(Into::into)
});
core.run(work).unwrap();
}

View File

@@ -66,6 +66,13 @@ pub struct Chunk {
inner: ::hyper::Chunk,
}
impl AsRef<[u8]> for Chunk {
#[inline]
fn as_ref(&self) -> &[u8] {
&*self
}
}
impl ::std::ops::Deref for Chunk {
type Target = [u8];
#[inline]

View File

@@ -82,6 +82,19 @@ impl fmt::Debug for Decoder {
}
impl Decoder {
/// An empty decoder.
///
/// This decoder will produce a single 0 byte chunk.
#[inline]
pub fn empty() -> Decoder {
Decoder {
inner: Inner::PlainText(body::empty())
}
}
/// A plain text decoder.
///
/// This decoder will emit the underlying chunks as-is.
#[inline]
fn plain_text(body: Body) -> Decoder {
Decoder {
@@ -89,6 +102,9 @@ impl Decoder {
}
}
/// A gzip decoder.
///
/// This decoder will buffer and decompress chunks that are gzipped.
#[inline]
fn gzip(mut body: Body) -> Decoder {
Decoder {
@@ -385,14 +401,6 @@ impl<S> ReadableChunks<S>
// pub(crate)
#[inline]
pub fn take(decoder: &mut Decoder) -> Decoder {
let inner = mem::replace(&mut decoder.inner, Inner::PlainText(body::empty()));
Decoder {
inner: inner,
}
}
/// Constructs a Decoder from a hyper request.
///
/// A decoder is just a wrapper around the hyper request that knows

View File

@@ -63,17 +63,17 @@ impl Response {
///
/// This function will replace the body on the response with an empty one.
#[inline]
pub fn body(&mut self) -> Decoder {
decoder::take(&mut self.body)
pub fn body(&self) -> &Decoder {
&self.body
}
/// Try to deserialize the response body as JSON using `serde`.
#[inline]
pub fn json<T: DeserializeOwned>(&mut self) -> Json<T> {
let body = self.body().concat2();
let body = mem::replace(&mut self.body, Decoder::empty());
Json {
concat: body,
concat: body.concat2(),
_marker: PhantomData,
}
}

View File

@@ -172,6 +172,7 @@ pub mod unstable {
pub use ::async_impl::{
Body,
Chunk,
Decoder,
Client,
ClientBuilder,
Request,

View File

@@ -1,3 +1,4 @@
use std::mem;
use std::fmt;
use std::io::{self, Read};
use std::time::Duration;
@@ -182,7 +183,7 @@ impl Response {
/// ```
#[inline]
pub fn error_for_status(self) -> ::Result<Self> {
let Response { inner, body, _thread_handle } = self;
let Response { body, inner, _thread_handle } = self;
inner.error_for_status().map(move |inner| {
Response {
inner: inner,
@@ -227,8 +228,9 @@ impl Stream for WaitBody {
// pub(crate)
pub fn new(mut res: async_impl::Response, timeout: Option<Duration>, thread: KeepCoreThreadAlive) -> Response {
let body = mem::replace(res.body_mut(), async_impl::Decoder::empty());
let body = async_impl::ReadableChunks::new(WaitBody {
inner: wait::stream(res.body(), timeout)
inner: wait::stream(body, timeout)
});
Response {

View File

@@ -9,6 +9,8 @@ extern crate libflate;
#[macro_use]
mod support;
use std::mem;
use reqwest::unstable::async::{Client, Decoder};
use futures::{Future, Stream};
use tokio_core::reactor::Core;
use std::io::Write;
@@ -59,12 +61,15 @@ fn test_gzip(response_size: usize, chunk_size: usize) {
let mut core = Core::new().unwrap();
let client = reqwest::unstable::async::Client::new(&core.handle()).unwrap();
let client = Client::new(&core.handle()).unwrap();
let res_future = client.get(&format!("http://{}/gzip", server.addr()))
.unwrap()
.send()
.and_then(|mut res| res.body().concat2())
.and_then(|mut res| {
let body = mem::replace(res.body_mut(), Decoder::empty());
body.concat2()
})
.and_then(|buf| {
let body = ::std::str::from_utf8(&buf).unwrap();