use failure::{Backtrace, Context, Fail};
use std::fmt::{self, Display};
use std::io;
#[derive(Debug)]
pub struct Error {
inner: Context<ErrorKind>,
}
#[derive(Fail, Debug)]
pub enum ErrorKind {
#[fail(display = "Call to pipe failed: {}", _0)]
Pipe(io::Error),
#[fail(display = "Couldn't fork daemon process: {}", _0)]
Fork(io::Error),
#[fail(display = "Couldn't redirect stdio streams: {}", _0)]
Dup2(io::Error),
#[fail(display = "Unable to create new session: {}", _0)]
DetachSession(io::Error),
#[fail(display = "Unable to change directory")]
ChangeDirectory,
#[fail(display = "pid_file option contains NUL")]
PathContainsNul,
#[fail(display = "Unable to open pid file, {}", _0)]
OpenPidfile(io::Error),
#[fail(display = "Unable to write self pid to pid file {}", _0)]
WritePid(io::Error),
#[fail(display = "Unable to register pipe with mio: {}", _0)]
RegisterationError(io::Error),
#[fail(display = "Unable to splice streams: {}", _0)]
SpliceError(io::Error),
#[fail(display = "Unable to fetch pending datasize from ioctl: {}", _0)]
Ioctl(io::Error),
#[fail(display = "{}", _0)]
Fnctl(io::Error),
#[fail(display = "Daemonize on the current platform is not supported")]
UnsupportedPlatform
}
impl Fail for Error {
fn cause(&self) -> Option<&Fail> {
self.inner.cause()
}
fn backtrace(&self) -> Option<&Backtrace> {
self.inner.backtrace()
}
}
impl Display for Error {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
Display::fmt(&self.inner, f)
}
}
impl Error {
pub fn kind(&self) -> &ErrorKind {
self.inner.get_context()
}
}
impl From<ErrorKind> for Error {
fn from(kind: ErrorKind) -> Error {
Error {
inner: Context::new(kind),
}
}
}
impl From<Context<ErrorKind>> for Error {
fn from(inner: Context<ErrorKind>) -> Error {
Error { inner }
}
}