#[cfg(feature = "alloc")]
use alloc::format;
use core::fmt;
use yansi::{
Color::{Fixed, Green, Red, Unset},
Style,
};
macro_rules! paint {
($f:expr, $colour:expr, $fmt:expr, $($args:tt)*) => (
write!($f, "{}", $colour.paint(format!($fmt, $($args)*)))
)
}
const SIGN_RIGHT: char = '>'; const SIGN_LEFT: char = '<'; pub(crate) fn write_header(f: &mut fmt::Formatter) -> fmt::Result {
writeln!(
f,
"{} {} / {} :",
Style::new(Unset).bold().paint("Diff"),
Red.paint(format!("{} left", SIGN_LEFT)),
Green.paint(format!("right {}", SIGN_RIGHT))
)
}
#[derive(Default)]
struct LatentDeletion<'a> {
value: Option<&'a str>,
count: usize,
}
impl<'a> LatentDeletion<'a> {
fn set(&mut self, value: &'a str) {
self.value = Some(value);
self.count += 1;
}
fn take(&mut self) -> Option<&'a str> {
if self.count == 1 {
self.value.take()
} else {
None
}
}
fn flush<TWrite: fmt::Write>(&mut self, f: &mut TWrite) -> fmt::Result {
if let Some(value) = self.value {
paint!(f, Red, "{}{}", SIGN_LEFT, value)?;
writeln!(f)?;
self.value = None;
} else {
self.count = 0;
}
Ok(())
}
}
pub(crate) fn write_lines<TWrite: fmt::Write>(
f: &mut TWrite,
left: &str,
right: &str,
) -> fmt::Result {
let diff = ::diff::lines(left, right);
let mut changes = diff.into_iter().peekable();
let mut previous_deletion = LatentDeletion::default();
while let Some(change) = changes.next() {
match (change, changes.peek()) {
(::diff::Result::Both(value, _), _) => {
previous_deletion.flush(f)?;
writeln!(f, " {}", value)?;
}
(::diff::Result::Left(deleted), _) => {
previous_deletion.flush(f)?;
previous_deletion.set(deleted);
}
(::diff::Result::Right(inserted), Some(::diff::Result::Right(_))) => {
previous_deletion.flush(f)?;
paint!(f, Green, "{}{}", SIGN_RIGHT, inserted)?;
writeln!(f)?;
}
(::diff::Result::Right(inserted), _) => {
if let Some(deleted) = previous_deletion.take() {
write_inline_diff(f, deleted, inserted)?;
} else {
previous_deletion.flush(f)?;
paint!(f, Green, "{}{}", SIGN_RIGHT, inserted)?;
writeln!(f)?;
}
}
};
}
previous_deletion.flush(f)?;
Ok(())
}
struct InlineWriter<'a, Writer> {
f: &'a mut Writer,
style: Style,
}
impl<'a, Writer> InlineWriter<'a, Writer>
where
Writer: fmt::Write,
{
fn new(f: &'a mut Writer) -> Self {
InlineWriter {
f,
style: Style::new(Unset),
}
}
fn write_with_style(&mut self, c: &char, style: Style) -> fmt::Result {
if style == self.style {
write!(self.f, "{}", c)?;
} else {
self.style.fmt_suffix(self.f)?;
style.fmt_prefix(self.f)?;
write!(self.f, "{}", c)?;
self.style = style;
}
Ok(())
}
fn finish(&mut self) -> fmt::Result {
self.style.fmt_suffix(self.f)?;
writeln!(self.f)?;
self.style = Style::new(Unset);
Ok(())
}
}
fn write_inline_diff<TWrite: fmt::Write>(f: &mut TWrite, left: &str, right: &str) -> fmt::Result {
let diff = ::diff::chars(left, right);
let mut writer = InlineWriter::new(f);
let light = Style::new(Red);
let heavy = Style::new(Red).bg(Fixed(52)).bold();
writer.write_with_style(&SIGN_LEFT, light)?;
for change in diff.iter() {
match change {
::diff::Result::Both(value, _) => writer.write_with_style(value, light)?,
::diff::Result::Left(value) => writer.write_with_style(value, heavy)?,
_ => (),
}
}
writer.finish()?;
let light = Style::new(Green);
let heavy = Style::new(Green).bg(Fixed(22)).bold();
writer.write_with_style(&SIGN_RIGHT, light)?;
for change in diff.iter() {
match change {
::diff::Result::Both(value, _) => writer.write_with_style(value, light)?,
::diff::Result::Right(value) => writer.write_with_style(value, heavy)?,
_ => (),
}
}
writer.finish()
}
#[cfg(test)]
mod test {
use super::*;
#[cfg(feature = "alloc")]
use alloc::string::String;
const RED_LIGHT: &str = "\u{1b}[31m";
const GREEN_LIGHT: &str = "\u{1b}[32m";
const RED_HEAVY: &str = "\u{1b}[1;48;5;52;31m";
const GREEN_HEAVY: &str = "\u{1b}[1;48;5;22;32m";
const RESET: &str = "\u{1b}[0m";
fn check_printer<TPrint>(printer: TPrint, left: &str, right: &str, expected: &str)
where
TPrint: Fn(&mut String, &str, &str) -> fmt::Result,
{
let mut actual = String::new();
printer(&mut actual, left, right).expect("printer function failed");
#[cfg(feature = "std")]
println!(
"## left ##\n\
{}\n\
## right ##\n\
{}\n\
## actual diff ##\n\
{}\n\
## expected diff ##\n\
{}",
left, right, actual, expected
);
assert_eq!(actual, expected);
}
#[test]
fn write_inline_diff_empty() {
let left = "";
let right = "";
let expected = format!(
"{red_light}<{reset}\n\
{green_light}>{reset}\n",
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_inline_diff, left, right, &expected);
}
#[test]
fn write_inline_diff_added() {
let left = "";
let right = "polymerase";
let expected = format!(
"{red_light}<{reset}\n\
{green_light}>{reset}{green_heavy}polymerase{reset}\n",
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
green_heavy = GREEN_HEAVY,
reset = RESET,
);
check_printer(write_inline_diff, left, right, &expected);
}
#[test]
fn write_inline_diff_removed() {
let left = "polyacrylamide";
let right = "";
let expected = format!(
"{red_light}<{reset}{red_heavy}polyacrylamide{reset}\n\
{green_light}>{reset}\n",
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
red_heavy = RED_HEAVY,
reset = RESET,
);
check_printer(write_inline_diff, left, right, &expected);
}
#[test]
fn write_inline_diff_changed() {
let left = "polymerase";
let right = "polyacrylamide";
let expected = format!(
"{red_light}<poly{reset}{red_heavy}me{reset}{red_light}ra{reset}{red_heavy}s{reset}{red_light}e{reset}\n\
{green_light}>poly{reset}{green_heavy}ac{reset}{green_light}r{reset}{green_heavy}yl{reset}{green_light}a{reset}{green_heavy}mid{reset}{green_light}e{reset}\n",
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
red_heavy = RED_HEAVY,
green_heavy = GREEN_HEAVY,
reset = RESET,
);
check_printer(write_inline_diff, left, right, &expected);
}
#[test]
fn write_lines_empty_string() {
let left = "";
let right = "content";
let expected = format!(
"{green_light}>content{reset}\n",
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn write_lines_struct() {
let left = r#"Some(
Foo {
lorem: "Hello World!",
ipsum: 42,
dolor: Ok(
"hey",
),
},
)"#;
let right = r#"Some(
Foo {
lorem: "Hello Wrold!",
ipsum: 42,
dolor: Ok(
"hey ho!",
),
},
)"#;
let expected = format!(
r#" Some(
Foo {{
{red_light}< lorem: "Hello W{reset}{red_heavy}o{reset}{red_light}rld!",{reset}
{green_light}> lorem: "Hello Wr{reset}{green_heavy}o{reset}{green_light}ld!",{reset}
ipsum: 42,
dolor: Ok(
{red_light}< "hey",{reset}
{green_light}> "hey{reset}{green_heavy} ho!{reset}{green_light}",{reset}
),
}},
)
"#,
red_light = RED_LIGHT,
red_heavy = RED_HEAVY,
green_light = GREEN_LIGHT,
green_heavy = GREEN_HEAVY,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn write_lines_multiline_block() {
let left = r#"Proboscis
Cabbage"#;
let right = r#"Probed
Caravaggio"#;
let expected = format!(
r#"{red_light}<Proboscis{reset}
{red_light}<Cabbage{reset}
{green_light}>Probed{reset}
{green_light}>Caravaggio{reset}
"#,
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn write_lines_multiline_insert() {
let left = r#"Cabbage"#;
let right = r#"Probed
Caravaggio"#;
let expected = format!(
r#"{red_light}<Cabbage{reset}
{green_light}>Probed{reset}
{green_light}>Caravaggio{reset}
"#,
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn write_lines_multiline_delete() {
let left = r#"Proboscis
Cabbage"#;
let right = r#"Probed"#;
let expected = format!(
r#"{red_light}<Proboscis{reset}
{red_light}<Cabbage{reset}
{green_light}>Probed{reset}
"#,
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn write_lines_issue12() {
let left = r#"[
0,
0,
0,
128,
10,
191,
5,
64,
]"#;
let right = r#"[
84,
248,
45,
64,
]"#;
let expected = format!(
r#" [
{red_light}< 0,{reset}
{red_light}< 0,{reset}
{red_light}< 0,{reset}
{red_light}< 128,{reset}
{red_light}< 10,{reset}
{red_light}< 191,{reset}
{red_light}< 5,{reset}
{green_light}> 84,{reset}
{green_light}> 248,{reset}
{green_light}> 45,{reset}
64,
]
"#,
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
mod write_lines_edge_newlines {
use super::*;
#[test]
fn both_trailing() {
let left = "fan\n";
let right = "mug\n";
let expected = format!(
r#"{red_light}<{reset}{red_heavy}fan{reset}
{green_light}>{reset}{green_heavy}mug{reset}
"#,
red_light = RED_LIGHT,
red_heavy = RED_HEAVY,
green_light = GREEN_LIGHT,
green_heavy = GREEN_HEAVY,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn both_leading() {
let left = "\nfan";
let right = "\nmug";
let expected = format!(
r#"
{red_light}<{reset}{red_heavy}fan{reset}
{green_light}>{reset}{green_heavy}mug{reset}
"#,
red_light = RED_LIGHT,
red_heavy = RED_HEAVY,
green_light = GREEN_LIGHT,
green_heavy = GREEN_HEAVY,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn leading_added() {
let left = "fan";
let right = "\nmug";
let expected = format!(
r#"{red_light}<fan{reset}
{green_light}>{reset}
{green_light}>mug{reset}
"#,
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn leading_deleted() {
let left = "\nfan";
let right = "mug";
let expected = format!(
r#"{red_light}<{reset}
{red_light}<fan{reset}
{green_light}>mug{reset}
"#,
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn trailing_added() {
let left = "fan";
let right = "mug\n";
let expected = format!(
r#"{red_light}<fan{reset}
{green_light}>mug{reset}
{green_light}>{reset}
"#,
red_light = RED_LIGHT,
green_light = GREEN_LIGHT,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
#[test]
fn trailing_deleted() {
let left = "fan\n";
let right = "mug";
let expected = format!(
r#"{red_light}<{reset}{red_heavy}fan{reset}
{green_light}>{reset}{green_heavy}mug{reset}
{red_light}<{reset}
"#,
red_light = RED_LIGHT,
red_heavy = RED_HEAVY,
green_light = GREEN_LIGHT,
green_heavy = GREEN_HEAVY,
reset = RESET,
);
check_printer(write_lines, left, right, &expected);
}
}
}