anvil_zksync_common/
shell.rsuse anstream::{AutoStream, ColorChoice as AnstreamColorChoice};
use anstyle::{AnsiColor, Effects, Reset, Style};
use std::fmt::Arguments;
use std::io::{self, IsTerminal, Write};
use std::sync::{Mutex, OnceLock};
pub const ERROR: Style = AnsiColor::Red.on_default().effects(Effects::BOLD);
pub const WARN: Style = AnsiColor::Yellow.on_default().effects(Effects::BOLD);
#[derive(Debug, Clone, Copy, PartialEq)]
pub enum ColorChoice {
Auto,
Always,
Never,
}
impl ColorChoice {
#[inline]
fn to_anstream(self) -> AnstreamColorChoice {
match self {
ColorChoice::Always => AnstreamColorChoice::Always,
ColorChoice::Never => AnstreamColorChoice::Never,
ColorChoice::Auto => AnstreamColorChoice::Auto,
}
}
}
#[derive(Debug, Clone, Copy, PartialEq)]
pub enum OutputMode {
Normal,
Quiet,
}
#[derive(Debug)]
pub struct Shell {
pub verbosity: u8,
pub output_mode: OutputMode,
pub color_choice: ColorChoice,
stdout: AutoStream<std::io::Stdout>,
stderr: AutoStream<std::io::Stderr>,
}
impl Shell {
pub fn new() -> Self {
let color = ColorChoice::Auto;
Self {
verbosity: 0,
output_mode: OutputMode::Normal,
color_choice: color,
stdout: AutoStream::new(std::io::stdout(), color.to_anstream()),
stderr: AutoStream::new(std::io::stderr(), color.to_anstream()),
}
}
pub fn print_out(&mut self, args: Arguments) -> io::Result<()> {
if self.output_mode == OutputMode::Quiet {
return Ok(());
}
self.stdout.write_fmt(args)?;
self.stdout.flush()
}
pub fn println_out(&mut self, args: Arguments) -> io::Result<()> {
if self.output_mode == OutputMode::Quiet {
return Ok(());
}
self.stdout.write_fmt(args)?;
writeln!(self.stdout)?;
self.stdout.flush()
}
pub fn print_err(&mut self, args: Arguments) -> io::Result<()> {
if self.output_mode == OutputMode::Quiet {
return Ok(());
}
self.stderr.write_fmt(args)?;
self.stderr.flush()
}
pub fn println_err(&mut self, args: Arguments) -> io::Result<()> {
if self.output_mode == OutputMode::Quiet {
return Ok(());
}
self.stderr.write_fmt(args)?;
writeln!(self.stderr)?;
self.stderr.flush()
}
pub fn warn(&mut self, args: Arguments) -> io::Result<()> {
if self.should_color() {
write!(self.stderr, "{}Warning:{} ", WARN, Reset)?;
} else {
write!(self.stderr, "Warning: ")?;
}
self.stderr.write_fmt(args)?;
writeln!(self.stderr)?;
self.stderr.flush()
}
pub fn error(&mut self, args: Arguments) -> io::Result<()> {
if self.should_color() {
write!(self.stderr, "{}Error:{} ", ERROR, Reset)?;
} else {
write!(self.stderr, "Error: ")?;
}
self.stderr.write_fmt(args)?;
writeln!(self.stderr)?;
self.stderr.flush()
}
fn should_color(&self) -> bool {
match self.color_choice {
ColorChoice::Always => true,
ColorChoice::Never => false,
ColorChoice::Auto => std::io::stdout().is_terminal(),
}
}
}
impl Default for Shell {
fn default() -> Self {
Self::new()
}
}
static GLOBAL_SHELL: OnceLock<Mutex<Shell>> = OnceLock::new();
pub fn get_shell() -> std::sync::MutexGuard<'static, Shell> {
GLOBAL_SHELL
.get_or_init(|| Mutex::new(Shell::new()))
.lock()
.expect("global shell mutex is poisoned")
}
pub fn set_shell(shell: Shell) {
let _ = GLOBAL_SHELL.set(Mutex::new(shell));
}
#[macro_export]
macro_rules! sh_print {
($($arg:tt)*) => {{
$crate::shell::get_shell().print_out(format_args!($($arg)*))
.unwrap_or_else(|e| eprintln!("Error writing output: {}", e));
}};
}
#[macro_export]
macro_rules! sh_println {
($($arg:tt)*) => {{
$crate::shell::get_shell().println_out(format_args!($($arg)*))
.unwrap_or_else(|e| eprintln!("Error writing output: {}", e));
}};
}
#[macro_export]
macro_rules! sh_eprint {
($($arg:tt)*) => {{
$crate::shell::get_shell().print_err(format_args!($($arg)*))
.unwrap_or_else(|e| eprintln!("Error writing stderr: {}", e));
}};
}
#[macro_export]
macro_rules! sh_eprintln {
($($arg:tt)*) => {{
$crate::shell::get_shell().println_err(format_args!($($arg)*))
.unwrap_or_else(|e| eprintln!("Error writing stderr: {}", e));
}};
}
#[macro_export]
macro_rules! sh_warn {
($($arg:tt)*) => {{
$crate::shell::get_shell().warn(format_args!($($arg)*))
.unwrap_or_else(|e| eprintln!("Error writing warning: {}", e));
}};
}
#[macro_export]
macro_rules! sh_err {
($($arg:tt)*) => {{
$crate::shell::get_shell().error(format_args!($($arg)*))
.unwrap_or_else(|e| eprintln!("Error writing error: {}", e));
}};
}
#[cfg(test)]
mod tests {
#[test]
fn test_shell_macros() {
sh_print!("Hello, ");
sh_println!("world!");
sh_eprint!("Error: ");
sh_eprintln!("Something went wrong!");
sh_warn!("This is a warning");
sh_err!("This is an error");
}
}