1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-05 05:49:20 -05:00

perf: don't allocate so much in runtime/colors.rs (#21957)

This commit is contained in:
Luca Casonato 2024-01-16 14:58:01 +01:00 committed by GitHub
parent b142c81721
commit 028f854468
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -2,8 +2,8 @@
use once_cell::sync::Lazy; use once_cell::sync::Lazy;
use std::fmt; use std::fmt;
use std::fmt::Write as _;
use std::io::IsTerminal; use std::io::IsTerminal;
use std::io::Write;
use termcolor::Ansi; use termcolor::Ansi;
use termcolor::Color::Ansi256; use termcolor::Color::Ansi256;
use termcolor::Color::Black; use termcolor::Color::Black;
@ -43,121 +43,181 @@ pub fn enable_ansi() {
BufferWriter::stdout(ColorChoice::AlwaysAnsi); BufferWriter::stdout(ColorChoice::AlwaysAnsi);
} }
fn style<S: AsRef<str>>(s: S, colorspec: ColorSpec) -> impl fmt::Display { /// A struct that can adapt a `fmt::Write` to a `std::io::Write`. If anything
if !use_color() { /// that can not be represented as UTF-8 is written to this writer, it will
return String::from(s.as_ref()); /// return an error.
struct StdFmtStdIoWriter<'a>(&'a mut dyn fmt::Write);
impl std::io::Write for StdFmtStdIoWriter<'_> {
fn write(&mut self, buf: &[u8]) -> std::io::Result<usize> {
let str = std::str::from_utf8(buf).map_err(|_| {
std::io::Error::new(
std::io::ErrorKind::Other,
"failed to convert bytes to str",
)
})?;
match self.0.write_str(str) {
Ok(_) => Ok(buf.len()),
Err(_) => Err(std::io::Error::new(
std::io::ErrorKind::Other,
"failed to write to fmt::Write",
)),
} }
let mut v = Vec::new();
let mut ansi_writer = Ansi::new(&mut v);
ansi_writer.set_color(&colorspec).unwrap();
ansi_writer.write_all(s.as_ref().as_bytes()).unwrap();
ansi_writer.reset().unwrap();
String::from_utf8_lossy(&v).into_owned()
} }
pub fn red_bold<S: AsRef<str>>(s: S) -> impl fmt::Display { fn flush(&mut self) -> std::io::Result<()> {
Ok(())
}
}
/// A struct that can adapt a `std::io::Write` to a `fmt::Write`.
struct StdIoStdFmtWriter<'a>(&'a mut dyn std::io::Write);
impl fmt::Write for StdIoStdFmtWriter<'_> {
fn write_str(&mut self, s: &str) -> fmt::Result {
self.0.write_all(s.as_bytes()).map_err(|_| fmt::Error)?;
Ok(())
}
}
struct Style<I: fmt::Display> {
colorspec: ColorSpec,
inner: I,
}
impl<I: fmt::Display> fmt::Display for Style<I> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if !use_color() {
return fmt::Display::fmt(&self.inner, f);
}
let mut ansi_writer = Ansi::new(StdFmtStdIoWriter(f));
ansi_writer
.set_color(&self.colorspec)
.map_err(|_| fmt::Error)?;
write!(StdIoStdFmtWriter(&mut ansi_writer), "{}", self.inner)?;
ansi_writer.reset().map_err(|_| fmt::Error)?;
Ok(())
}
}
#[inline]
fn style<'a>(
s: impl fmt::Display + 'a,
colorspec: ColorSpec,
) -> impl fmt::Display + 'a {
Style {
colorspec,
inner: s,
}
}
pub fn red_bold<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Red)).set_bold(true); style_spec.set_fg(Some(Red)).set_bold(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn green_bold<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn green_bold<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Green)).set_bold(true); style_spec.set_fg(Some(Green)).set_bold(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn italic<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn italic<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_italic(true); style_spec.set_italic(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn italic_gray<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn italic_gray<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Ansi256(8))).set_italic(true); style_spec.set_fg(Some(Ansi256(8))).set_italic(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn italic_bold<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn italic_bold<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_bold(true).set_italic(true); style_spec.set_bold(true).set_italic(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn white_on_red<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn white_on_red<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_bg(Some(Red)).set_fg(Some(White)); style_spec.set_bg(Some(Red)).set_fg(Some(White));
style(s, style_spec) style(s, style_spec)
} }
pub fn black_on_green<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn black_on_green<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_bg(Some(Green)).set_fg(Some(Black)); style_spec.set_bg(Some(Green)).set_fg(Some(Black));
style(s, style_spec) style(s, style_spec)
} }
pub fn yellow<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn yellow<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Yellow)); style_spec.set_fg(Some(Yellow));
style(s, style_spec) style(s, style_spec)
} }
pub fn cyan<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn cyan<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Cyan)); style_spec.set_fg(Some(Cyan));
style(s, style_spec) style(s, style_spec)
} }
pub fn cyan_with_underline<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn cyan_with_underline<'a>(
s: impl fmt::Display + 'a,
) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Cyan)).set_underline(true); style_spec.set_fg(Some(Cyan)).set_underline(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn cyan_bold<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn cyan_bold<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Cyan)).set_bold(true); style_spec.set_fg(Some(Cyan)).set_bold(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn magenta<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn magenta<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Magenta)); style_spec.set_fg(Some(Magenta));
style(s, style_spec) style(s, style_spec)
} }
pub fn red<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn red<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Red)); style_spec.set_fg(Some(Red));
style(s, style_spec) style(s, style_spec)
} }
pub fn green<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn green<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Green)); style_spec.set_fg(Some(Green));
style(s, style_spec) style(s, style_spec)
} }
pub fn bold<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn bold<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_bold(true); style_spec.set_bold(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn gray<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn gray<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Ansi256(245))); style_spec.set_fg(Some(Ansi256(245)));
style(s, style_spec) style(s, style_spec)
} }
pub fn intense_blue<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn intense_blue<'a>(s: impl fmt::Display + 'a) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec.set_fg(Some(Blue)).set_intense(true); style_spec.set_fg(Some(Blue)).set_intense(true);
style(s, style_spec) style(s, style_spec)
} }
pub fn white_bold_on_red<S: AsRef<str>>(s: S) -> impl fmt::Display { pub fn white_bold_on_red<'a>(
s: impl fmt::Display + 'a,
) -> impl fmt::Display + 'a {
let mut style_spec = ColorSpec::new(); let mut style_spec = ColorSpec::new();
style_spec style_spec
.set_bold(true) .set_bold(true)