1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
// Copyright 2020 Palantir Technologies, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
use crate::blocking::runtime;
use crate::raw::DefaultRawBody;
use bytes::Bytes;
use futures::executor;
use futures::future;
use http_body::Body;
use hyper::{HeaderMap, StatusCode};
use std::error;
use std::io::{self, BufRead, Read};
use std::pin::Pin;
use tokio::io::{AsyncBufRead, AsyncReadExt};

/// A blocking HTTP response.
pub struct Response<B = DefaultRawBody>(crate::Response<B>);

impl<B> Response<B> {
    pub(crate) fn new(inner: crate::Response<B>) -> Response<B> {
        Response(inner)
    }

    /// Returns the response's status.
    pub fn status(&self) -> StatusCode {
        self.0.status()
    }

    /// Returns the response's headers.
    pub fn headers(&self) -> &HeaderMap {
        self.0.headers()
    }

    /// Consumes the response, returning its body.
    pub fn into_body(self) -> ResponseBody<B> {
        ResponseBody(Box::pin(self.0.into_body()))
    }
}

/// A blocking streaming response body.
pub struct ResponseBody<B = DefaultRawBody>(Pin<Box<crate::ResponseBody<B>>>);

impl<B> ResponseBody<B>
where
    B: Body<Data = Bytes>,
    B::Error: Into<Box<dyn error::Error + Sync + Send>>,
{
    /// Reads the next chunk of bytes from the response.
    ///
    /// Compared to the `Read` implementation, this method avoids some copies of the body data when working with an API
    /// that already consumes `Bytes` objects.
    pub fn read_bytes(&mut self) -> io::Result<Option<Bytes>> {
        let _guard = runtime()?.enter();
        executor::block_on(self.0.as_mut().read_bytes())
    }
}

impl<B> Read for ResponseBody<B>
where
    B: Body<Data = Bytes>,
    B::Error: Into<Box<dyn error::Error + Sync + Send>>,
{
    fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
        let _guard = runtime()?.enter();
        executor::block_on(self.0.read(buf))
    }
}

impl<B> BufRead for ResponseBody<B>
where
    B: Body<Data = Bytes>,
    B::Error: Into<Box<dyn error::Error + Sync + Send>>,
{
    fn fill_buf(&mut self) -> io::Result<&[u8]> {
        // lifetime shenanigans mean we can't return the value of poll_fill_buf directly
        let _guard = runtime()?.enter();
        executor::block_on(future::poll_fn(|cx| {
            self.0.as_mut().poll_fill_buf(cx).map_ok(|_| ())
        }))?;
        Ok(self.0.buffer())
    }

    fn consume(&mut self, n: usize) {
        self.0.as_mut().consume(n)
    }
}