flexi_logger 0.22.3

An easy-to-configure and flexible logger that writes logs to stderr or stdout and/or to files. It allows custom logline formats, and it allows changing the log specification at runtime. It also allows defining additional log streams, e.g. for alert or security messages.
Documentation
//! ```cargo
//! [dependencies]
//! yansi = "0.5"
//! ```
extern crate yansi;
use std::process::Command;

macro_rules! run_command {
    ($cmd:expr , $($arg:expr),*) => (
        let mut command = command!($cmd, $($arg),*);
        let mut child = command.spawn().unwrap();
        let status = child.wait().unwrap();
        if !status.success() {
            print!("> {}",yansi::Paint::red("qualify terminates due to error"));
            std::process::exit(-1);
        }
    )
}

macro_rules! command {
    ($cmd:expr , $($arg:expr),*) => (
        {
            print!("\n> {}",yansi::Paint::yellow($cmd));
            let mut command = Command::new($cmd);
            $(
                print!(" {}",yansi::Paint::yellow(&$arg));
                command.arg($arg);
            )*
            print!("\n");
            command
        }
    )
}

fn run_script(s: &str) {
    let mut path = std::path::PathBuf::from(std::env::var("CARGO_SCRIPT_BASE_PATH").unwrap());
    path.push(s);
    let script = path.to_string_lossy().to_owned().to_string();
    run_command!("cargo", "script", script);
}

fn main() {
    println!("Qualify flexi_logger");

    // format
    run_command!("cargo", "fmt");

    // Build in important variants
    std::fs::remove_file("Cargo.lock").ok();
    run_command!("cargo", "+1.53.0", "build", "--no-default-features");
    run_command!("cargo", "+1.53.0", "build", "--all-features");

    std::fs::remove_file("Cargo.lock").ok();
    run_command!("cargo", "build");
    run_command!("cargo", "build", "--no-default-features");
    run_command!("cargo", "build", "--no-default-features", "--features=atty");
    run_command!("cargo", "build", "--all-features");
    run_command!("cargo", "build", "--release");
    run_command!("cargo", "build", "--release", "--all-features");

    // Clippy in important variants
    run_command!("cargo", "clippy", "--", "-D", "warnings");
    run_command!("cargo", "clippy", "--all-features", "--", "-D", "warnings");
    #[rustfmt::skip]
    run_command!("cargo", "+nightly", "clippy", "--all-targets", "--all-features", "--", "-D", "warnings");

    // Run tests in important variants
    run_command!("cargo", "+1.53.0", "test", "--all-features");
    run_command!("cargo", "test", "--release", "--all-features");
    run_command!("cargo", "test", "--no-default-features");
    run_command!("cargo", "test", "--release");
    #[rustfmt::skip]
    run_command!("cargo", "test", "--release", "--features", "specfile_without_notification");

    // doc
    run_command!("cargo", "+nightly", "test", "--all-features", "--doc");
    #[rustfmt::skip]
    run_command!("cargo", "+nightly", "doc", "--all-features", "--no-deps", "--open");

    // check version consistency
    run_command!("cargo", "run", "--example", "version_numbers");

    // check git status
    let mut cmd = command!("git", "status", "-s");
    let child = cmd.stdout(std::process::Stdio::piped()).spawn().unwrap();
    let output = child.wait_with_output().unwrap();
    if output.stdout.len() > 0 {
        print!("> {}", yansi::Paint::red("there are unsubmitted files"));
        std::process::exit(-1);
    }
    // say goodbye
    println!(
        "\n\
    > all done :-)  Looks like you're ready to\n\
    - \"git push\"\n\
    - check if the github actions were successful, and then\n\
    - \"cargo publish\""
    );

    // cleanup
    run_script("cleanup");
}