async-h1 2.1.3

Asynchronous HTTP 1.1 parser.
Documentation
use std::fmt;
use std::pin::Pin;
use std::task::{Context, Poll};

use async_std::io::{self, BufRead, Read};
use async_std::sync::Sender;

pin_project_lite::pin_project! {
    /// ReadNotifier forwards [`async_std::io::Read`] and
    /// [`async_std::io::BufRead`] to an inner reader. When the
    /// ReadNotifier is read from (using `Read`, `ReadExt`, or
    /// `BufRead` methods), it sends a single message containing `()`
    /// on the channel.
    pub(crate) struct ReadNotifier<B> {
        #[pin]
        reader: B,
        sender: Sender<()>,
        has_been_read: bool
    }
}

impl<B> fmt::Debug for ReadNotifier<B> {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        f.debug_struct("ReadNotifier")
            .field("read", &self.has_been_read)
            .finish()
    }
}

impl<B: BufRead> ReadNotifier<B> {
    pub(crate) fn new(reader: B, sender: Sender<()>) -> Self {
        Self {
            reader,
            sender,
            has_been_read: false,
        }
    }
}

impl<B: BufRead> BufRead for ReadNotifier<B> {
    fn poll_fill_buf(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<&[u8]>> {
        self.project().reader.poll_fill_buf(cx)
    }

    fn consume(self: Pin<&mut Self>, amt: usize) {
        self.project().reader.consume(amt)
    }
}

impl<B: Read> Read for ReadNotifier<B> {
    fn poll_read(
        self: Pin<&mut Self>,
        cx: &mut Context<'_>,
        buf: &mut [u8],
    ) -> Poll<io::Result<usize>> {
        let this = self.project();

        if !*this.has_been_read {
            if let Ok(()) = this.sender.try_send(()) {
                *this.has_been_read = true;
            };
        }

        this.reader.poll_read(cx, buf)
    }
}