Implement LineWriter and BufWriter

This commit is contained in:
Kirill Mironov 2019-08-20 19:06:20 +03:00
parent 48d4c9b18d
commit a0759a6c53
2 changed files with 269 additions and 80 deletions

View file

@ -1,9 +1,8 @@
use crate::task::{Context, Poll}; use crate::task::{Context, Poll};
use futures::{ready, AsyncWrite, Future, Stream}; use futures::{ready, AsyncWrite};
use std::io::{self, IntoInnerError};
use std::pin::Pin;
use std::fmt; use std::fmt;
use crate::io::Write; use std::io;
use std::pin::Pin;
const DEFAULT_CAPACITY: usize = 8 * 1024; const DEFAULT_CAPACITY: usize = 8 * 1024;
@ -34,41 +33,50 @@ const DEFAULT_CAPACITY: usize = 8 * 1024;
/// ///
/// Let's write the numbers one through ten to a [`TcpStream`]: /// Let's write the numbers one through ten to a [`TcpStream`]:
/// ///
/*/ ```no_run /// ```no_run
/ use std::io::prelude::*; /// # #![feature(async_await)]
/ use std::net::TcpStream; /// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/ /// use async_std::net::TcpStream;
/ let mut stream = TcpStream::connect("127.0.0.1:34254").unwrap(); /// use async_std::io::Write;
/ ///
/ for i in 0..10 { /// let mut stream = TcpStream::connect("127.0.0.1:34254").await?;
/ stream.write(&[i+1]).unwrap(); ///
/ } /// for i in 0..10 {
/ ```*/ /// let arr = [i+1];
/// stream.write(&arr).await?;
/// }
/// #
/// # Ok(()) }) }
/// ```
/// ///
/// Because we're not buffering, we write each one in turn, incurring the /// Because we're not buffering, we write each one in turn, incurring the
/// overhead of a system call per byte written. We can fix this with a /// overhead of a system call per byte written. We can fix this with a
/// `BufWriter`: /// `BufWriter`:
/// ///
/*/ ```no_run /// ```no_run
/ use std::io::prelude::*; /// # #![feature(async_await)]
/ use std::io::BufWriter; /// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/ use std::net::TcpStream; /// use async_std::io::BufWriter;
/ /// use async_std::net::TcpStream;
/ let mut stream = BufWriter::new(TcpStream::connect("127.0.0.1:34254").unwrap()); /// use async_std::io::Write;
/ ///
/ for i in 0..10 { /// let mut stream = BufWriter::new(TcpStream::connect("127.0.0.1:34254").await?);
/ stream.write(&[i+1]).unwrap(); /// for i in 0..10 {
/ } /// let arr = [i+1];
/ ```*/ /// stream.write(&arr).await?;
/// };
/// #
/// # Ok(()) }) }
/// ```
/// ///
/// By wrapping the stream with a `BufWriter`, these ten writes are all grouped /// By wrapping the stream with a `BufWriter`, these ten writes are all grouped
/// together by the buffer, and will all be written out in one system call when /// together by the buffer, and will all be written out in one system call when
/// the `stream` is dropped. /// the `stream` is dropped.
/// ///
/// [`Write`]: ../../std/io/trait.Write.html /// [`Write`]: trait.Write.html
/// [`TcpStream::write`]: ../../std/net/struct.TcpStream.html#method.write /// [`TcpStream::write`]: ../net/struct.TcpStream.html#method.write
/// [`TcpStream`]: ../../std/net/struct.TcpStream.html /// [`TcpStream`]: ../net/struct.TcpStream.html
/// [`flush`]: #method.flush /// [`flush`]: trait.Write.html#tymethod.flush
pub struct BufWriter<W> { pub struct BufWriter<W> {
inner: W, inner: W,
buf: Vec<u8>, buf: Vec<u8>,
@ -85,10 +93,15 @@ impl<W: AsyncWrite + Unpin> BufWriter<W> {
/// # Examples /// # Examples
/// ///
/// ```no_run /// ```no_run
/// # #![feature(async_await)]
/// # #![allow(unused_mut)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::BufWriter; /// use async_std::io::BufWriter;
/// use async_std::net::TcpStream; /// use async_std::net::TcpStream;
/// ///
/// let mut buffer = BufWriter::new(TcpStream::connect("127.0.0.1:34254").unwrap()); /// let mut buffer = BufWriter::new(TcpStream::connect("127.0.0.1:34254").await?);
/// #
/// # Ok(()) }) }
/// ``` /// ```
pub fn new(inner: W) -> BufWriter<W> { pub fn new(inner: W) -> BufWriter<W> {
BufWriter::with_capacity(DEFAULT_CAPACITY, inner) BufWriter::with_capacity(DEFAULT_CAPACITY, inner)
@ -101,11 +114,16 @@ impl<W: AsyncWrite + Unpin> BufWriter<W> {
/// Creating a buffer with a buffer of a hundred bytes. /// Creating a buffer with a buffer of a hundred bytes.
/// ///
/// ```no_run /// ```no_run
/// # #![feature(async_await)]
/// # #![allow(unused_mut)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::BufWriter; /// use async_std::io::BufWriter;
/// use async_std::net::TcpStream; /// use async_std::net::TcpStream;
/// ///
/// let stream = TcpStream::connect("127.0.0.1:34254").unwrap(); /// let stream = TcpStream::connect("127.0.0.1:34254").await?;
/// let mut buffer = BufWriter::with_capacity(100, stream); /// let mut buffer = BufWriter::with_capacity(100, stream);
/// #
/// # Ok(()) }) }
/// ``` /// ```
pub fn with_capacity(capacity: usize, inner: W) -> BufWriter<W> { pub fn with_capacity(capacity: usize, inner: W) -> BufWriter<W> {
BufWriter { BufWriter {
@ -115,17 +133,54 @@ impl<W: AsyncWrite + Unpin> BufWriter<W> {
} }
} }
/// Gets a reference to the underlying writer.
///
/// # Examples
///
/// ```no_run
/// # #![feature(async_await)]
/// # #![allow(unused_mut)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::BufWriter;
/// use async_std::net::TcpStream;
///
/// let mut buffer = BufWriter::new(TcpStream::connect("127.0.0.1:34254").await?);
///
/// // We can use reference just like buffer
/// let reference = buffer.get_ref();
/// #
/// # Ok(()) }) }
/// ```
pub fn get_ref(&self) -> &W { pub fn get_ref(&self) -> &W {
&self.inner &self.inner
} }
/// Gets a mutable reference to the underlying writer.
///
/// It is inadvisable to directly write to the underlying writer.
///
/// # Examples
///
/// ```no_run
/// # #![feature(async_await)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::BufWriter;
/// use async_std::net::TcpStream;
///
/// let mut buffer = BufWriter::new(TcpStream::connect("127.0.0.1:34254").await?);
///
/// // We can use reference just like buffer
/// let reference = buffer.get_mut();
/// #
/// # Ok(()) }) }
/// ```
pub fn get_mut(&mut self) -> &mut W { pub fn get_mut(&mut self) -> &mut W {
&mut self.inner &mut self.inner
} }
pub fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut W> { // pub fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut W> {
self.inner() // self.inner()
} // }
/// Consumes BufWriter, returning the underlying writer /// Consumes BufWriter, returning the underlying writer
/// ///
@ -137,19 +192,41 @@ impl<W: AsyncWrite + Unpin> BufWriter<W> {
self.inner self.inner
} }
pub fn poll_into_inner(mut self: Pin<&mut Self>, cx: Context<'_>) -> Poll<io::Result<usize>> { // pub fn poll_into_inner(self: Pin<&mut Self>, _cx: Context<'_>) -> Poll<io::Result<usize>> {
unimplemented!("poll into inner method") // unimplemented!("poll into inner method")
} // }
/// Returns a reference to the internally buffered data.
///
/// # Examples
///
/// ```no_run
/// # #![feature(async_await)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::BufWriter;
/// use async_std::net::TcpStream;
///
/// let buf_writer = BufWriter::new(TcpStream::connect("127.0.0.1:34251").await?);
///
/// // See how many bytes are currently buffered
/// let bytes_buffered = buf_writer.buffer().len();
/// #
/// # Ok(()) }) }
/// ```
pub fn buffer(&self) -> &[u8] { pub fn buffer(&self) -> &[u8] {
&self.buf &self.buf
} }
pub fn poll_flush_buf(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> { /// Poll buffer flushing until completion
///
/// This is used in types that wrap around BufWrite, one such example: [`LineWriter`]
///
/// [`LineWriter`]: struct.LineWriter.html
pub fn poll_flush_buf(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
let Self { let Self {
inner, inner,
buf, buf,
written written,
} = Pin::get_mut(self); } = Pin::get_mut(self);
let mut inner = Pin::new(inner); let mut inner = Pin::new(inner);
let len = buf.len(); let len = buf.len();
@ -183,7 +260,7 @@ impl<W: AsyncWrite + Unpin> BufWriter<W> {
impl<W: AsyncWrite + Unpin> AsyncWrite for BufWriter<W> { impl<W: AsyncWrite + Unpin> AsyncWrite for BufWriter<W> {
fn poll_write( fn poll_write(
mut self: Pin<&mut Self>, mut self: Pin<&mut Self>,
cx: &mut Context, cx: &mut Context<'_>,
buf: &[u8], buf: &[u8],
) -> Poll<io::Result<usize>> { ) -> Poll<io::Result<usize>> {
if self.buf.len() + buf.len() > self.buf.capacity() { if self.buf.len() + buf.len() > self.buf.capacity() {
@ -192,16 +269,16 @@ impl<W: AsyncWrite + Unpin> AsyncWrite for BufWriter<W> {
if buf.len() >= self.buf.capacity() { if buf.len() >= self.buf.capacity() {
self.inner().poll_write(cx, buf) self.inner().poll_write(cx, buf)
} else { } else {
self.buf().write(buf).poll() Pin::new(&mut *self.buf()).poll_write(cx, buf)
} }
} }
fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context) -> Poll<io::Result<()>> { fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
ready!(self.as_mut().poll_flush_buf(cx))?; ready!(self.as_mut().poll_flush_buf(cx))?;
self.inner().poll_flush(cx) self.inner().poll_flush(cx)
} }
fn poll_close(mut self: Pin<&mut Self>, cx: &mut Context) -> Poll<io::Result<()>> { fn poll_close(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
ready!(self.as_mut().poll_flush_buf(cx))?; ready!(self.as_mut().poll_flush_buf(cx))?;
self.inner().poll_close(cx) self.inner().poll_close(cx)
} }
@ -211,14 +288,78 @@ impl<W: AsyncWrite + fmt::Debug + Unpin> fmt::Debug for BufWriter<W> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("BufReader") f.debug_struct("BufReader")
.field("writer", &self.inner) .field("writer", &self.inner)
.field( .field("buf", &self.buf)
"buf",
&self.buf
)
.finish() .finish()
} }
} }
/// Wraps a writer and buffers output to it, flushing whenever a newline
/// (`0x0a`, `'\n'`) is detected.
///
/// The [`BufWriter`][bufwriter] struct wraps a writer and buffers its output.
/// But it only does this batched write when it goes out of scope, or when the
/// internal buffer is full. Sometimes, you'd prefer to write each line as it's
/// completed, rather than the entire buffer at once. Enter `LineWriter`. It
/// does exactly that.
///
/// Like [`BufWriter`][bufwriter], a `LineWriter`s buffer will also be flushed when the
/// `LineWriter` goes out of scope or when its internal buffer is full.
///
/// [bufwriter]: struct.BufWriter.html
///
/// If there's still a partial line in the buffer when the `LineWriter` is
/// dropped, it will flush those contents.
///
/// This type is an async version of [`std::io::LineWriter`]
///
/// [`std::io::LineWriter`]: https://doc.rust-lang.org/std/io/struct.LineWriter.html
///
/// # Examples
///
/// We can use `LineWriter` to write one line at a time, significantly
/// reducing the number of actual writes to the file.
///
/// ```no_run
/// # #![feature(async_await)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::{LineWriter, Write};
/// use async_std::fs::{File, self};
/// let road_not_taken = b"I shall be telling this with a sigh
/// Somewhere ages and ages hence:
/// Two roads diverged in a wood, and I -
/// I took the one less traveled by,
/// And that has made all the difference.";
///
/// let file = File::create("poem.txt").await?;
/// let mut file = LineWriter::new(file);
///
/// file.write_all(b"I shall be telling this with a sigh").await?;
///
/// // No bytes are written until a newline is encountered (or
/// // the internal buffer is filled).
/// assert_eq!(fs::read_to_string("poem.txt").await?, "");
/// file.write_all(b"\n").await?;
/// assert_eq!(
/// fs::read_to_string("poem.txt").await?,
/// "I shall be telling this with a sigh\n",
/// );
///
/// // Write the rest of the poem.
/// file.write_all(b"Somewhere ages and ages hence:
/// Two roads diverged in a wood, and I -
/// I took the one less traveled by,
/// And that has made all the difference.").await?;
///
/// // The last line of the poem doesn't end in a newline, so
/// // we have to flush or drop the `LineWriter` to finish
/// // writing.
/// file.flush().await?;
///
/// // Confirm the whole poem was written.
/// assert_eq!(fs::read("poem.txt").await?, &road_not_taken[..]);
/// #
/// # Ok(()) }) }
/// ```
pub struct LineWriter<W: AsyncWrite + Unpin> { pub struct LineWriter<W: AsyncWrite + Unpin> {
inner: BufWriter<W>, inner: BufWriter<W>,
need_flush: bool, need_flush: bool,
@ -232,16 +373,15 @@ impl<W: AsyncWrite + Unpin> LineWriter<W> {
/// # Examples /// # Examples
/// ///
/// ```no_run /// ```no_run
/// # #![feature(async_await)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::fs::File; /// use async_std::fs::File;
/// use async_std::io::LineWriter; /// use async_std::io::LineWriter;
/// ///
/// fn main() -> std::io::Result<()> {
/// async_std::task::block_on(async {
/// let file = File::create("poem.txt").await?; /// let file = File::create("poem.txt").await?;
/// let file = LineWriter::new(file); /// let file = LineWriter::new(file);
/// Ok(()) /// #
/// }) /// # Ok(()) }) }
/// }
/// ``` /// ```
pub fn new(inner: W) -> LineWriter<W> { pub fn new(inner: W) -> LineWriter<W> {
// Lines typically aren't that long, don't use a giant buffer // Lines typically aren't that long, don't use a giant buffer
@ -254,16 +394,15 @@ impl<W: AsyncWrite + Unpin> LineWriter<W> {
/// # Examples /// # Examples
/// ///
/// ```no_run /// ```no_run
/// # #![feature(async_await)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::fs::File; /// use async_std::fs::File;
/// use async_std::io::LineWriter; /// use async_std::io::LineWriter;
/// ///
/// fn main() -> std::io::Result<()> {
/// async_std::task::block_on(async {
/// let file = File::create("poem.txt").await?; /// let file = File::create("poem.txt").await?;
/// let file = LineWriter::with_capacity(100, file); /// let file = LineWriter::with_capacity(100, file);
/// Ok(()) /// #
/// }) /// # Ok(()) }) }
/// }
/// ``` /// ```
pub fn with_capacity(capacity: usize, inner: W) -> LineWriter<W> { pub fn with_capacity(capacity: usize, inner: W) -> LineWriter<W> {
LineWriter { LineWriter {
@ -272,68 +411,120 @@ impl<W: AsyncWrite + Unpin> LineWriter<W> {
} }
} }
/// Gets a reference to the underlying writer.
///
/// # Examples
///
/// ```no_run
/// # #![feature(async_await)]
/// # #![allow(unused_mut)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::LineWriter;
/// use async_std::fs::File;
///
/// let file = File::create("poem.txt").await?;
/// let file = LineWriter::new(file);
///
/// // We can use reference just like buffer
/// let reference = file.get_ref();
/// #
/// # Ok(()) }) }
/// ```
pub fn get_ref(&self) -> &W { pub fn get_ref(&self) -> &W {
self.inner.get_ref() self.inner.get_ref()
} }
/// Gets a mutable reference to the underlying writer.
///
/// Caution must be taken when calling methods on the mutable reference
/// returned as extra writes could corrupt the output stream.
///
/// # Examples
///
/// ```no_run
/// # #![feature(async_await)]
/// # #![allow(unused_mut)]
/// # fn main() -> std::io::Result<()> { async_std::task::block_on(async {
/// use async_std::io::LineWriter;
/// use async_std::fs::File;
///
/// let file = File::create("poem.txt").await?;
/// let mut file = LineWriter::new(file);
///
/// // We can use reference just like buffer
/// let reference = file.get_mut();
/// #
/// # Ok(()) }) }
/// ```
pub fn get_mut(&mut self) -> &mut W { pub fn get_mut(&mut self) -> &mut W {
self.inner.get_mut() self.inner.get_mut()
} }
//TODO: Implement flushing before returning inner
#[allow(missing_docs)]
pub fn into_inner(self) -> W { pub fn into_inner(self) -> W {
self.inner.into_inner() self.inner.into_inner()
} }
} }
impl<W: AsyncWrite + Unpin> AsyncWrite for LineWriter<W> { impl<W: AsyncWrite + Unpin> AsyncWrite for LineWriter<W> {
fn poll_write(mut self: Pin<&mut Self>, cx: &mut Context, buf: &[u8]) -> Poll<io::Result<usize>> { fn poll_write(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &[u8],
) -> Poll<io::Result<usize>> {
if self.need_flush { if self.need_flush {
self.as_mut().poll_flush(cx)?; let _ = self.as_mut().poll_flush(cx)?;
} }
let i = match memchr::memrchr(b'\n', buf) { let i = match memchr::memrchr(b'\n', buf) {
Some(i) => i, Some(i) => i,
None => return self.as_mut().inner().as_mut().poll_write(cx, buf) None => return self.as_mut().inner().as_mut().poll_write(cx, buf),
}; };
let n = ready!(self.as_mut().inner().as_mut().poll_write(cx, &buf[..=i])?); let n = ready!(self.as_mut().inner().as_mut().poll_write(cx, &buf[..=i])?);
*self.as_mut().need_flush() = true; *self.as_mut().need_flush() = true;
if ready!(self.as_mut().poll_flush(cx)).is_err() || n != 1 + 1 { if ready!(self.as_mut().poll_flush(cx)).is_err() || n != i + 1 {
return Poll::Ready(Ok(n)) return Poll::Ready(Ok(n));
} }
match ready!(self.inner().poll_write(cx, &buf[i + 1..])) { match ready!(self.inner().poll_write(cx, &buf[i + 1..])) {
Ok(i) => Poll::Ready(Ok(n + 1)), Ok(_) => Poll::Ready(Ok(n + 1)),
Err(_) => Poll::Ready(Ok(n)) Err(_) => Poll::Ready(Ok(n)),
} }
} }
fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context) -> Poll<io::Result<()>> { fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
self.as_mut().inner().poll_flush(cx)?; let _ = self.as_mut().inner().poll_flush(cx)?;
*self.need_flush() = false; *self.need_flush() = false;
Poll::Ready(Ok(())) Poll::Ready(Ok(()))
} }
fn poll_close(mut self: Pin<&mut Self>, cx: &mut Context) -> Poll<io::Result<()>> { fn poll_close(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<io::Result<()>> {
self.as_mut().inner().poll_flush(cx)?; let _ = self.as_mut().inner().poll_flush(cx)?;
self.inner().poll_close(cx) self.inner().poll_close(cx)
} }
} }
impl<W: AsyncWrite + Unpin> fmt::Debug for LineWriter<W> where W: fmt::Debug { impl<W: AsyncWrite + Unpin> fmt::Debug for LineWriter<W>
where
W: fmt::Debug,
{
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt.debug_struct("LineWriter") fmt.debug_struct("LineWriter")
.field("writer", &self.inner.inner) .field("writer", &self.inner.inner)
.field("buffer", .field(
&format_args!("{}/{}", self.inner.buf.len(), self.inner.buf.capacity())) "buffer",
&format_args!("{}/{}", self.inner.buf.len(), self.inner.buf.capacity()),
)
.finish() .finish()
} }
} }
mod tests { mod tests {
#![allow(unused_imports)]
use super::LineWriter;
use crate::prelude::*; use crate::prelude::*;
use crate::task; use crate::task;
use super::LineWriter;
#[test] #[test]
fn test_line_buffer() { fn test_line_buffer() {

View file

@ -21,8 +21,6 @@
//! # Ok(()) }) } //! # Ok(()) }) }
//! ``` //! ```
pub(crate) const DEFAULT_CAPACITY: usize = 8 * 1024;
#[doc(inline)] #[doc(inline)]
pub use std::io::{Error, ErrorKind, Result, SeekFrom}; pub use std::io::{Error, ErrorKind, Result, SeekFrom};