pub trait Write {
type Error;
fn write_all(&mut self, buf: &[u8]) -> Result<(), Self::Error>;
}
impl<W: Write + ?Sized> Write for &mut W {
type Error = W::Error;
fn write_all(&mut self, buf: &[u8]) -> Result<(), Self::Error> {
(**self).write_all(buf)
}
}
impl Write for &mut [u8] {
type Error = EndOfSlice;
fn write_all(&mut self, buf: &[u8]) -> Result<(), Self::Error> {
if self.len() < buf.len() {
return Err(EndOfSlice(()))
}
let this = core::mem::take(self);
let (prefix, suffix) = this.split_at_mut(buf.len());
prefix.copy_from_slice(buf);
*self = suffix;
Ok(())
}
}
#[cfg(feature = "alloc")]
impl Write for alloc::vec::Vec<u8> {
type Error = core::convert::Infallible;
fn write_all(&mut self, buf: &[u8]) -> Result<(), Self::Error> {
self.extend_from_slice(buf);
Ok(())
}
}
#[derive(Debug)]
pub struct Cursor<W> {
wrt: W,
pos: usize
}
impl<W> Cursor<W> {
pub fn new(w: W) -> Self {
Cursor { wrt: w, pos: 0 }
}
pub fn position(&self) -> usize {
self.pos
}
pub fn set_position(&mut self, pos: usize) {
self.pos = pos
}
pub fn get_ref(&self) -> &W {
&self.wrt
}
pub fn get_mut(&mut self) -> &mut W {
&mut self.wrt
}
pub fn into_inner(self) -> W {
self.wrt
}
}
impl Write for Cursor<&mut [u8]> {
type Error = EndOfSlice;
fn write_all(&mut self, buf: &[u8]) -> Result<(), Self::Error> {
let Some(slice) = self.wrt.get_mut(self.pos .. self.pos + buf.len()) else {
return Err(EndOfSlice(()))
};
slice.copy_from_slice(buf);
self.pos += buf.len();
Ok(())
}
}
impl<const N: usize> Write for Cursor<[u8; N]> {
type Error = EndOfArray;
fn write_all(&mut self, buf: &[u8]) -> Result<(), Self::Error> {
let Some(slice) = self.wrt.get_mut(self.pos .. self.pos + buf.len()) else {
return Err(EndOfArray(()))
};
slice.copy_from_slice(buf);
self.pos += buf.len();
Ok(())
}
}
#[cfg(feature = "alloc")]
impl Write for Cursor<alloc::boxed::Box<[u8]>> {
type Error = EndOfSlice;
fn write_all(&mut self, buf: &[u8]) -> Result<(), Self::Error> {
let Some(slice) = self.wrt.get_mut(self.pos .. self.pos + buf.len()) else {
return Err(EndOfSlice(()))
};
slice.copy_from_slice(buf);
self.pos += buf.len();
Ok(())
}
}
#[cfg(feature = "std")]
#[derive(Debug)]
pub struct Writer<W>(W);
#[cfg(feature = "std")]
impl<W> Writer<W> {
pub fn new(w: W) -> Self {
Writer(w)
}
pub fn get_ref(&self) -> &W {
&self.0
}
pub fn get_mut(&mut self) -> &mut W {
&mut self.0
}
pub fn into_inner(self) -> W {
self.0
}
}
#[cfg(feature = "std")]
impl<W: std::io::Write> Write for Writer<W> {
type Error = std::io::Error;
fn write_all(&mut self, buf: &[u8]) -> std::io::Result<()> {
std::io::Write::write_all(&mut self.0, buf)
}
}
#[derive(Debug)]
pub struct EndOfSlice(());
impl core::fmt::Display for EndOfSlice {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
f.write_str("end of slice")
}
}
impl core::error::Error for EndOfSlice {}
#[derive(Debug)]
pub struct EndOfArray(());
impl core::fmt::Display for EndOfArray {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
f.write_str("end of array")
}
}
impl core::error::Error for EndOfArray {}