use std::{error::Error as StdError, fmt};
pub type Result<T> = std::result::Result<T, Error>;
#[derive(Debug)]
pub struct Error {
kind: ErrorKind,
source: Option<Box<dyn StdError + Sync + Send + 'static>>,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum ErrorKind {
Io,
Parse,
InvalidArgument,
InvalidOperation,
}
impl StdError for Error {
fn source(&self) -> Option<&(dyn StdError + 'static)> {
match self.source {
Some(ref x) => Some(&**x),
None => None,
}
}
}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_str(match self.kind {
ErrorKind::Io => "Unable to do an I/O operation on a cgroup file system",
ErrorKind::Parse => "Unable to parse a content of a cgroup file",
ErrorKind::InvalidArgument => "Invalid argument",
ErrorKind::InvalidOperation => "Invalid operation",
})?;
if let Some(ref source) = self.source {
write!(f, ": {}", source)?;
}
Ok(())
}
}
impl Error {
pub(crate) fn new(kind: ErrorKind) -> Self {
Self { kind, source: None }
}
pub(crate) fn with_source<E>(kind: ErrorKind, source: E) -> Self
where
E: StdError + Sync + Send + 'static,
{
Self {
kind,
source: Some(Box::new(source)),
}
}
pub(crate) fn parse<E>(source: E) -> Self
where
E: StdError + Sync + Send + 'static,
{
Self::with_source(ErrorKind::Parse, source)
}
pub fn kind(&self) -> ErrorKind {
self.kind
}
}
impl From<std::io::Error> for Error {
fn from(e: std::io::Error) -> Self {
Self::with_source(ErrorKind::Io, e)
}
}
impl From<std::num::ParseIntError> for Error {
fn from(e: std::num::ParseIntError) -> Self {
Self::with_source(ErrorKind::Parse, e)
}
}
#[cfg(test)]
#[allow(unreachable_code, dead_code)]
fn test_error_impl_sync_send() {
let _e: Error = unimplemented!();
let _: &dyn Sync = &_e;
let _: &dyn Send = &_e;
}