use std::time::{SystemTime, SystemTimeError, UNIX_EPOCH};
use super::*;
#[derive(Debug, Clone)]
pub enum SystemTimeTransformError {
EncodeBufferTooSmall,
NotEnoughBytes,
InvalidSystemTime(SystemTimeError),
}
impl core::fmt::Display for SystemTimeTransformError {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
match self {
Self::EncodeBufferTooSmall => write!(
f,
"buffer is too small, use `Transformable::encoded_len` to pre-allocate a buffer with enough space"
),
Self::NotEnoughBytes => write!(f, "not enough bytes to decode system time"),
Self::InvalidSystemTime(e) => write!(f, "{e}"),
}
}
}
#[cfg(feature = "std")]
impl std::error::Error for SystemTimeTransformError {}
impl Transformable for SystemTime {
type Error = SystemTimeTransformError;
fn encode(&self, dst: &mut [u8]) -> Result<usize, Self::Error> {
if dst.len() < self.encoded_len() {
return Err(Self::Error::EncodeBufferTooSmall);
}
let buf = encode_duration_unchecked(
self
.duration_since(UNIX_EPOCH)
.map_err(Self::Error::InvalidSystemTime)?,
);
dst[..ENCODED_LEN].copy_from_slice(&buf);
Ok(ENCODED_LEN)
}
#[cfg(feature = "std")]
#[cfg_attr(docsrs, doc(cfg(feature = "std")))]
fn encode_to_writer<W: std::io::Write>(&self, writer: &mut W) -> std::io::Result<usize> {
let mut buf = [0u8; ENCODED_LEN];
self
.encode(&mut buf)
.map_err(|e| std::io::Error::new(std::io::ErrorKind::InvalidData, e))?;
writer.write_all(&buf).map(|_| ENCODED_LEN)
}
#[cfg(feature = "async")]
#[cfg_attr(docsrs, doc(cfg(feature = "async")))]
async fn encode_to_async_writer<W: futures_util::io::AsyncWrite + Send + Unpin>(
&self,
writer: &mut W,
) -> std::io::Result<usize>
where
Self::Error: Send + Sync + 'static,
{
use futures_util::AsyncWriteExt;
let mut buf = [0u8; ENCODED_LEN];
self
.encode(&mut buf)
.map_err(|e| std::io::Error::new(std::io::ErrorKind::InvalidData, e))?;
writer.write_all(&buf).await.map(|_| ENCODED_LEN)
}
fn encoded_len(&self) -> usize {
ENCODED_LEN
}
fn decode(src: &[u8]) -> Result<(usize, Self), Self::Error>
where
Self: Sized,
{
if src.len() < ENCODED_LEN {
return Err(Self::Error::NotEnoughBytes);
}
let (readed, dur) = decode_duration_unchecked(src);
Ok((readed, UNIX_EPOCH + dur))
}
#[cfg(feature = "std")]
#[cfg_attr(docsrs, doc(cfg(feature = "std")))]
fn decode_from_reader<R: std::io::Read>(reader: &mut R) -> std::io::Result<(usize, Self)>
where
Self: Sized,
{
let mut buf = [0; ENCODED_LEN];
reader.read_exact(&mut buf)?;
let (readed, dur) = decode_duration_unchecked(&buf);
Ok((readed, UNIX_EPOCH + dur))
}
#[cfg(feature = "async")]
#[cfg_attr(docsrs, doc(cfg(feature = "async")))]
async fn decode_from_async_reader<R: futures_util::io::AsyncRead + Send + Unpin>(
reader: &mut R,
) -> std::io::Result<(usize, Self)>
where
Self: Sized,
Self::Error: Send + Sync + 'static,
{
use futures_util::AsyncReadExt;
let mut buf = [0; ENCODED_LEN];
reader.read_exact(&mut buf).await?;
let (readed, dur) = decode_duration_unchecked(&buf);
Ok((readed, UNIX_EPOCH + dur))
}
}
test_transformable!(SystemTime => test_systemtime_transformable({
let now = SystemTime::now();
std::thread::sleep(std::time::Duration::from_millis(10));
now
}));