ia-sandbox 0.4.0

A CLI to sandbox (jail) and collect usage of applications.
Documentation
#![allow(clippy::use_self)]

use std::fmt::{self, Display, Formatter};
use std::time::Duration;

use serde::{Deserialize, Serialize};

use crate::config::{Limits, SpaceUsage};
use crate::utils::DurationDisplay;

#[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize, Deserialize)]
pub enum RunInfoResult<T> {
    Success(T),
    NonZeroExitStatus(u32),
    KilledBySignal(u32),
    MemoryLimitExceeded,
    TimeLimitExceeded,
    WallTimeLimitExceeded,
}

impl<T> RunInfoResult<T> {
    pub fn is_success(&self) -> bool {
        matches!(self, Self::Success(_))
    }

    pub fn and_then<A, B, F: FnOnce(T) -> Result<A, B>>(
        self,
        cb: F,
    ) -> Result<RunInfoResult<A>, B> {
        Ok(match self {
            Self::Success(obj) => RunInfoResult::Success(cb(obj)?),
            Self::NonZeroExitStatus(exit_status) => RunInfoResult::NonZeroExitStatus(exit_status),
            Self::KilledBySignal(signal) => RunInfoResult::KilledBySignal(signal),
            Self::MemoryLimitExceeded => RunInfoResult::MemoryLimitExceeded,
            Self::TimeLimitExceeded => RunInfoResult::TimeLimitExceeded,
            Self::WallTimeLimitExceeded => RunInfoResult::WallTimeLimitExceeded,
        })
    }

    pub fn success(self) -> Option<T> {
        match self {
            Self::Success(obj) => Some(obj),
            _ => None,
        }
    }
}

impl<T> Display for RunInfoResult<T> {
    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
        match *self {
            Self::Success(_) => write!(f, "Success"),
            Self::NonZeroExitStatus(ref exit_code) => {
                write!(f, "Non zero exit status: {}", exit_code)
            }
            Self::KilledBySignal(ref signal) => write!(f, "Killed by Signal {}", signal),
            Self::MemoryLimitExceeded => write!(f, "Memory limit exceeded"),
            Self::TimeLimitExceeded => write!(f, "Time limit exceeded"),
            Self::WallTimeLimitExceeded => write!(f, "Wall time limit exceeded"),
        }
    }
}

#[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize, Deserialize)]
pub struct RunUsage {
    user_time: Duration,
    wall_time: Duration,
    memory: SpaceUsage,
}

impl RunUsage {
    pub fn new(user_time: Duration, wall_time: Duration, memory: SpaceUsage) -> Self {
        Self {
            user_time,
            wall_time,
            memory,
        }
    }

    pub fn user_time(&self) -> Duration {
        self.user_time
    }

    pub fn wall_time(&self) -> Duration {
        self.wall_time
    }

    pub fn memory(&self) -> SpaceUsage {
        self.memory
    }

    pub fn check_limits<T>(self, limits: Limits) -> Option<RunInfo<T>> {
        if limits
            .user_time()
            .map_or(false, |time| time < self.user_time())
        {
            return Some(RunInfo::new(RunInfoResult::TimeLimitExceeded, self));
        }

        if limits
            .wall_time()
            .map_or(false, |time| time < self.wall_time())
        {
            return Some(RunInfo::new(RunInfoResult::WallTimeLimitExceeded, self));
        }

        if limits
            .memory()
            .map_or(false, |memory| memory < self.memory())
        {
            return Some(RunInfo::new(RunInfoResult::MemoryLimitExceeded, self));
        }

        None
    }
}

impl Default for RunUsage {
    fn default() -> Self {
        Self::new(
            Duration::from_secs(0),
            Duration::from_secs(0),
            SpaceUsage::from_bytes(0),
        )
    }
}

impl Display for RunUsage {
    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
        writeln!(f, "Total user time: {}", DurationDisplay(self.user_time()))?;
        writeln!(f, "Wall time: {}", DurationDisplay(self.wall_time()))?;
        write!(f, "Maximum memory: {}", self.memory())
    }
}
#[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize, Deserialize)]
pub struct RunInfo<T> {
    result: RunInfoResult<T>,
    usage: RunUsage,
}

#[allow(clippy::use_self)]
impl<T> RunInfo<T> {
    pub fn new(result: RunInfoResult<T>, usage: RunUsage) -> Self {
        Self { result, usage }
    }

    pub fn result(&self) -> &RunInfoResult<T> {
        &self.result
    }

    pub fn usage(&self) -> &RunUsage {
        &self.usage
    }

    pub fn is_success(&self) -> bool {
        self.result.is_success()
    }

    pub fn and_then<A, B, F: FnOnce(T) -> Result<A, B>>(self, cb: F) -> Result<RunInfo<A>, B> {
        let Self { result, usage } = self;
        result
            .and_then(cb)
            .map(|result| RunInfo::new(result, usage))
    }

    pub fn success(self) -> Option<T> {
        self.result.success()
    }
}

impl<T> Display for RunInfo<T> {
    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
        writeln!(f, "{}", self.result)?;
        write!(f, "{}", self.usage)
    }
}