forked from mirror/async-std
Implement LineWriter and BufWriter
This commit is contained in:
parent
48d4c9b18d
commit
a0759a6c53
2 changed files with 269 additions and 80 deletions
|
@ -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<()> {
|
/// let file = File::create("poem.txt").await?;
|
||||||
/// async_std::task::block_on(async {
|
/// let file = LineWriter::new(file);
|
||||||
/// let file = File::create("poem.txt").await?;
|
/// #
|
||||||
/// 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<()> {
|
/// let file = File::create("poem.txt").await?;
|
||||||
/// async_std::task::block_on(async {
|
/// let file = LineWriter::with_capacity(100, file);
|
||||||
/// let file = File::create("poem.txt").await?;
|
/// #
|
||||||
/// 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() {
|
||||||
|
|
|
@ -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};
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue