abscissa_core 0.9.0

Application microframework with support for command-line option parsing, configuration, error handling, logging, and terminal interactions. This crate contains the framework's core functionality.
Documentation
//! Terminal status handling.
//!
//! Presently provides a Cargo-like visual style. Hopefully in future versions
//! this can be made configurable.
//!
//! # `status_ok!`: Successful status messages
//!
//! ```ignore
//! // Print a Cargo-like justified status to STDOUT
//! status_ok!("Loaded", "app loaded successfully");
//! ```
//!
//! # `status_err!`: Error messages
//!
//! ```ignore
//! // Print an error message
//! status_err!("something bad happened");
//! ```
//!
//! # `status_attr_ok!`: Successful attributes
//!
//! ```ignore
//! // Print an indented attribute to STDOUT
//! status_attr_ok!("good", "yep");
//! ```
//!
//! # `status_attr_error!`: Error attributes
//!
//! ```ignore
//! // Print an error attribute to STDERR
//! status_attr_err!("error", "yep");
//! ```

use super::{stderr, stdout};
use crate::FrameworkError;
use std::io::Write;
use termcolor::{Color, ColorSpec, StandardStream, WriteColor};

/// Print a success status message (in green if colors are enabled)
///
/// ```ignore
/// // Print a Cargo-like justified status to STDOUT
/// status_ok!("Loaded", "app loaded successfully");
/// ```
#[macro_export]
macro_rules! status_ok {
    ($status:expr, $msg:expr) => {
        $crate::terminal::status::Status::new()
            .justified()
            .bold()
            .color($crate::terminal::Color::Green)
            .status($status)
            .print_stderr($msg)
            .unwrap();
    };
    ($status:expr, $fmt:expr, $($arg:tt)+) => {
        $crate::status_ok!($status, format!($fmt, $($arg)+));
    };
}

/// Print an informational status message (in cyan if colors are enabled)
///
/// ```ignore
/// // Print a Cargo-like justified status to STDOUT
/// status_info!("Info", "you may care to know about");
/// ```
#[macro_export]
macro_rules! status_info {
    ($status:expr, $msg:expr) => {
        $crate::terminal::status::Status::new()
            .justified()
            .bold()
            .color($crate::terminal::Color::Cyan)
            .status($status)
            .print_stderr($msg)
            .unwrap();
    };
    ($status:expr, $fmt:expr, $($arg:tt)+) => {
        $crate::status_info!($status, format!($fmt, $($arg)+));
    };
}

/// Print a warning status message (in yellow if colors are enabled)
///
/// ```ignore
/// // Print a Cargo-like justified status to STDOUT
/// status_warn!("heads up, there's something you should know");
/// ```
#[macro_export]
macro_rules! status_warn {
    ($msg:expr) => {
        $crate::terminal::status::Status::new()
            .bold()
            .color($crate::terminal::Color::Yellow)
            .status("warning:")
            .print_stderr($msg)
            .unwrap();
    };
    ($fmt:expr, $($arg:tt)+) => {
        $crate::status_warn!(format!($fmt, $($arg)+));
    };
}

/// Print an error message (in red if colors are enabled)
///
/// ```ignore
/// // Print an error message
/// status_err!("something bad happened");
/// ```
#[macro_export]
macro_rules! status_err {
    ($msg:expr) => {
        $crate::terminal::status::Status::new()
            .bold()
            .color($crate::terminal::Color::Red)
            .status("error:")
            .print_stderr($msg)
            .unwrap();
    };
    ($fmt:expr, $($arg:tt)+) => {
        $crate::status_err!(format!($fmt, $($arg)+));
    };
}

/// Print a tab-delimited status attribute (in green if colors are enabled)
///
/// ```ignore
/// // Print an indented attribute to STDOUT
/// status_attr_ok!("good", "yep");
/// ```
#[macro_export]
macro_rules! status_attr_ok {
    ($attr:expr, $msg:expr) => {
        // TODO(tarcieri): hax... use a better format string?
        let attr_delimited = if $attr.len() >= 7 {
            format!("{}:", $attr)
        } else {
            format!("{}:\t", $attr)
        };


        $crate::terminal::status::Status::new()
            .bold()
            .color($crate::terminal::Color::Green)
            .status(attr_delimited)
            .print_stdout($msg)
            .unwrap();
    };
    ($attr: expr, $fmt:expr, $($arg:tt)+) => {
        $crate::status_attr_ok!($attr, format!($fmt, $($arg)+));
    }
}

/// Print a tab-delimited status attribute (in red if colors are enabled)
///
/// ```ignore
/// // Print an error attribute to STDERR
/// status_attr_err!("error", "yep");
/// ```
#[macro_export]
macro_rules! status_attr_err {
    ($attr:expr, $msg:expr) => {
        // TODO(tarcieri): hax... use a better format string?
        let attr_delimited = if $attr.len() >= 7 {
            format!("{}:", $attr)
        } else {
            format!("{}:\t", $attr)
        };


        $crate::terminal::status::Status::new()
            .bold()
            .color($crate::terminal::Color::Red)
            .status(attr_delimited)
            .print_stdout($msg)
            .unwrap();
    };
    ($attr: expr, $fmt:expr, $($arg:tt)+) => {
        $crate::status_attr_err!($attr, format!($fmt, $($arg)+));
    }
}

/// Status message builder
#[derive(Clone, Debug, Default)]
pub struct Status {
    /// Should the status be justified?
    justified: bool,

    /// Should colors be bold?
    bold: bool,

    /// Color in which status should be displayed
    color: Option<Color>,

    /// Prefix of the status message (e.g. `Success`)
    status: Option<String>,
}

impl Status {
    /// Create a new status message with default settings
    pub fn new() -> Self {
        Self::default()
    }

    /// Justify status on display
    pub fn justified(mut self) -> Self {
        self.justified = true;
        self
    }

    /// Make colors bold
    pub fn bold(mut self) -> Self {
        self.bold = true;
        self
    }

    /// Set the colors used to display this message
    pub fn color(mut self, c: Color) -> Self {
        self.color = Some(c);
        self
    }

    /// Set a status message to display
    pub fn status<S>(mut self, msg: S) -> Self
    where
        S: ToString,
    {
        self.status = Some(msg.to_string());
        self
    }

    /// Print the given message to stdout
    pub fn print_stdout<S>(self, msg: S) -> Result<(), FrameworkError>
    where
        S: AsRef<str>,
    {
        self.print(stdout(), msg)
    }

    /// Print the given message to stderr
    pub fn print_stderr<S>(self, msg: S) -> Result<(), FrameworkError>
    where
        S: AsRef<str>,
    {
        self.print(stderr(), msg)
    }

    /// Print the given message
    fn print<S>(self, stream: &StandardStream, msg: S) -> Result<(), FrameworkError>
    where
        S: AsRef<str>,
    {
        let mut s = stream.lock();
        s.reset()?;
        s.set_color(ColorSpec::new().set_fg(self.color).set_bold(self.bold))?;

        if let Some(status) = self.status {
            if self.justified {
                write!(s, "{:>12}", status)?;
            } else {
                write!(s, "{}", status)?;
            }
        }

        s.reset()?;
        let msg = msg.as_ref();
        if !msg.is_empty() {
            writeln!(s, " {}", msg)?;
        }
        s.flush()?;

        Ok(())
    }
}