mockito 1.7.2

HTTP mocking for Rust.
Documentation
use crate::error::Error;
use crate::Request;
use bytes::Bytes;
use futures_core::Stream;
use http::{HeaderMap, StatusCode};
use std::fmt;
use std::io;
use std::sync::Arc;
use std::task::Poll;
use std::thread;
use tokio::sync::mpsc;

#[derive(Clone, Debug, PartialEq)]
pub(crate) struct Response {
    pub status: ResponseStatusCode,
    pub headers: HeaderMap<Header>,
    pub body: Body,
}

#[derive(Clone)]
pub(crate) enum ResponseStatusCode {
    StatusCode(StatusCode),
    FnWithRequest(Arc<StatusCodeFnWithRequest>),
}

impl fmt::Debug for ResponseStatusCode {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match *self {
            ResponseStatusCode::StatusCode(ref s) => s.fmt(f),
            ResponseStatusCode::FnWithRequest(_) => f.write_str("<callback>"),
        }
    }
}

impl PartialEq for ResponseStatusCode {
    fn eq(&self, other: &Self) -> bool {
        match (self, other) {
            (ResponseStatusCode::StatusCode(ref a), ResponseStatusCode::StatusCode(ref b)) => {
                a == b
            }
            (
                ResponseStatusCode::FnWithRequest(ref a),
                ResponseStatusCode::FnWithRequest(ref b),
            ) => std::ptr::eq(
                a.as_ref() as *const StatusCodeFnWithRequest as *const u8,
                b.as_ref() as *const StatusCodeFnWithRequest as *const u8,
            ),
            _ => false,
        }
    }
}

#[derive(Clone)]
pub(crate) enum Header {
    String(String),
    FnWithRequest(Arc<HeaderFnWithRequest>),
}

impl fmt::Debug for Header {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match *self {
            Header::String(ref s) => s.fmt(f),
            Header::FnWithRequest(_) => f.write_str("<callback>"),
        }
    }
}

impl PartialEq for Header {
    fn eq(&self, other: &Self) -> bool {
        match (self, other) {
            (Header::String(ref a), Header::String(ref b)) => a == b,
            (Header::FnWithRequest(ref a), Header::FnWithRequest(ref b)) => std::ptr::eq(
                a.as_ref() as *const HeaderFnWithRequest as *const u8,
                b.as_ref() as *const HeaderFnWithRequest as *const u8,
            ),
            _ => false,
        }
    }
}

type StatusCodeFnWithRequest = dyn Fn(&Request) -> usize + Send + Sync;
type HeaderFnWithRequest = dyn Fn(&Request) -> String + Send + Sync;

type BodyFnWithWriter = dyn Fn(&mut dyn io::Write) -> io::Result<()> + Send + Sync + 'static;
type BodyFnWithRequest = dyn Fn(&Request) -> Bytes + Send + Sync + 'static;

#[derive(Clone)]
pub(crate) enum Body {
    Bytes(Bytes),
    FnWithWriter(Arc<BodyFnWithWriter>),
    FnWithRequest(Arc<BodyFnWithRequest>),
}

impl fmt::Debug for Body {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match *self {
            Body::Bytes(ref b) => b.fmt(f),
            Body::FnWithWriter(_) => f.write_str("<callback>"),
            Body::FnWithRequest(_) => f.write_str("<callback>"),
        }
    }
}

impl PartialEq for Body {
    fn eq(&self, other: &Self) -> bool {
        match (self, other) {
            (Body::Bytes(ref a), Body::Bytes(ref b)) => a == b,
            (Body::FnWithWriter(ref a), Body::FnWithWriter(ref b)) => std::ptr::eq(
                a.as_ref() as *const BodyFnWithWriter as *const u8,
                b.as_ref() as *const BodyFnWithWriter as *const u8,
            ),
            (Body::FnWithRequest(ref a), Body::FnWithRequest(ref b)) => std::ptr::eq(
                a.as_ref() as *const BodyFnWithRequest as *const u8,
                b.as_ref() as *const BodyFnWithRequest as *const u8,
            ),
            _ => false,
        }
    }
}

impl Default for Response {
    fn default() -> Self {
        let mut headers = HeaderMap::with_capacity(1);
        headers.insert("connection", Header::String("close".to_string()));
        Self {
            status: ResponseStatusCode::StatusCode(StatusCode::OK),
            headers,
            body: Body::Bytes(Bytes::new()),
        }
    }
}

struct ChunkedStreamWriter {
    sender: mpsc::Sender<io::Result<Box<[u8]>>>,
}

impl io::Write for ChunkedStreamWriter {
    fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
        self.sender
            .blocking_send(Ok(buf.into()))
            .map_err(|_| io::ErrorKind::BrokenPipe)?;
        Ok(buf.len())
    }

    fn flush(&mut self) -> io::Result<()> {
        Ok(())
    }
}

pub(crate) struct ChunkedStream {
    receiver: Option<mpsc::Receiver<io::Result<Box<[u8]>>>>,
    thread: Option<thread::JoinHandle<()>>,
}

impl ChunkedStream {
    pub fn new(body_fn: Arc<BodyFnWithWriter>) -> Result<Self, Error> {
        let (sender, receiver) = mpsc::channel(1);
        let join = thread::Builder::new()
            .name(format!("mockito::body_fn_{:p}", body_fn))
            .spawn(move || {
                let mut writer = ChunkedStreamWriter { sender };
                if let Err(e) = body_fn(&mut writer) {
                    let _ = writer.sender.blocking_send(Err(e));
                }
            })
            .map_err(|e| Error::new_with_context(crate::ErrorKind::ResponseFailure, e))?;
        Ok(Self {
            receiver: Some(receiver),
            thread: Some(join),
        })
    }
}

impl Drop for ChunkedStream {
    fn drop(&mut self) {
        // must close the channel first
        let _ = self.receiver.take();
        let _ = self.thread.take().map(|t| t.join());
    }
}

impl Stream for ChunkedStream {
    type Item = io::Result<Bytes>;

    fn poll_next(
        mut self: std::pin::Pin<&mut Self>,
        cx: &mut std::task::Context<'_>,
    ) -> Poll<Option<Self::Item>> {
        self.receiver
            .as_mut()
            .map(move |receiver| {
                receiver
                    .poll_recv(cx)
                    .map(|received| received.map(|result| result.map(Into::into)))
            })
            .unwrap_or(Poll::Ready(None))
    }
}