From b671e391db9e5a05421fbd3a4e4ec86c095b49af Mon Sep 17 00:00:00 2001 From: Lauchmelder Date: Mon, 21 Nov 2022 23:05:54 +0100 Subject: [PATCH] finished the thing --- .vscode/launch.json | 45 +++++++++++++++ Cargo.lock | 18 ++++++ Cargo.toml | 1 + res/file1.txt | 1 + res/file2.txt | 1 + src/diff.rs | 135 +++++++++++++++++++++++++++++++++++++++++--- src/grid.rs | 37 ++++++++---- src/main.rs | 6 ++ 8 files changed, 225 insertions(+), 19 deletions(-) create mode 100644 .vscode/launch.json create mode 100644 res/file1.txt create mode 100644 res/file2.txt diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..4b7a357 --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,45 @@ +{ + // Use IntelliSense to learn about possible attributes. + // Hover to view descriptions of existing attributes. + // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 + "version": "0.2.0", + "configurations": [ + { + "type": "lldb", + "request": "launch", + "name": "Debug executable 'diff-tool'", + "cargo": { + "args": [ + "build", + "--bin=diff-tool", + "--package=diff-tool" + ], + "filter": { + "name": "diff-tool", + "kind": "bin" + } + }, + "args": ["--", "res/file1.txt", "res/file2.txt"], + "cwd": "${workspaceFolder}" + }, + { + "type": "lldb", + "request": "launch", + "name": "Debug unit tests in executable 'diff-tool'", + "cargo": { + "args": [ + "test", + "--no-run", + "--bin=diff-tool", + "--package=diff-tool" + ], + "filter": { + "name": "diff-tool", + "kind": "bin" + } + }, + "args": [], + "cwd": "${workspaceFolder}" + } + ] +} \ No newline at end of file diff --git a/Cargo.lock b/Cargo.lock index a0e93bb..4553993 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -54,12 +54,24 @@ dependencies = [ "os_str_bytes", ] +[[package]] +name = "colored" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b3616f750b84d8f0de8a58bda93e08e2a81ad3f523089b05f1dffecab48c6cbd" +dependencies = [ + "atty", + "lazy_static", + "winapi", +] + [[package]] name = "diff-tool" version = "0.1.0" dependencies = [ "clap", "clap_derive", + "colored", ] [[package]] @@ -77,6 +89,12 @@ dependencies = [ "libc", ] +[[package]] +name = "lazy_static" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" + [[package]] name = "libc" version = "0.2.137" diff --git a/Cargo.toml b/Cargo.toml index 5e0836b..e978571 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -8,3 +8,4 @@ edition = "2021" [dependencies] clap = "4.0.26" clap_derive = "4.0.21" +colored = "2.0.0" diff --git a/res/file1.txt b/res/file1.txt new file mode 100644 index 0000000..a3fe17f --- /dev/null +++ b/res/file1.txt @@ -0,0 +1 @@ +This is not a text file \ No newline at end of file diff --git a/res/file2.txt b/res/file2.txt new file mode 100644 index 0000000..815f461 --- /dev/null +++ b/res/file2.txt @@ -0,0 +1 @@ +This is a text file tho \ No newline at end of file diff --git a/src/diff.rs b/src/diff.rs index 88d1de0..02e53f7 100644 --- a/src/diff.rs +++ b/src/diff.rs @@ -1,6 +1,6 @@ -use std::{fs::File, io::{BufReader, Read}}; - +use std::{fs::File, io::{BufReader, Read}, fmt::Display}; use crate::grid::Grid; +use colored::*; macro_rules! safe_unwrap { ($expression: expr) => { @@ -11,13 +11,32 @@ macro_rules! safe_unwrap { }; } -#[derive(Default, Clone)] +#[derive(Clone)] enum Arrow { - #[default] LEFT, + NONE, + LEFT, UP, DIAGONAL } +impl Display for Arrow { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match self { + Arrow::NONE => write!(f, " "), + Arrow::LEFT => write!(f, "<-"), + Arrow::UP => write!(f, "^^"), + Arrow::DIAGONAL => write!(f, "<^") + } + } +} + +#[derive(Debug)] +enum Action { + INSERT(u32, char), + DELETE(u32), + REPLACE(usize, char) +} + fn open_and_read(file: String) -> Result { let fp = File::open(file)?; let mut buf_reader = BufReader::new(fp); @@ -27,9 +46,21 @@ fn open_and_read(file: String) -> Result { Ok(content) } +#[derive(Clone)] +struct Cell(usize, Arrow); + +impl Display for Cell { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + write!(f, "({},{})", self.0, self.1) + } +} + +#[derive(Debug)] pub struct Diff { first: String, second: String, + + actions: Vec } impl Diff { @@ -39,15 +70,105 @@ impl Diff { let mut diff = Diff { first: first_string, - second: second_string + second: second_string, + + actions: vec![] }; + let grid = diff.create_lcs(); + let mut pos = (diff.first.len() as u32, diff.second.len() as u32); + while pos != (0, 0) { + pos = match grid[pos].1 { + Arrow::DIAGONAL => (pos.0 - 1, pos.1 - 1), + Arrow::UP => { + diff.actions.push(Action::INSERT(pos.0, diff.second.chars().nth(pos.1 as usize - 1).unwrap())); + (pos.0, pos.1 - 1) + }, + Arrow::LEFT => { + diff.actions.push(Action::DELETE(pos.0)); + (pos.0 - 1, pos.1) + }, + Arrow::NONE => { + break; + } + } + } Ok(diff) } - fn create_lcs(&self) -> Grid { - Grid::new(self.first.len() as u32, self.second.len() as u32) + pub fn get_input(&self) -> (&String, &String) { + (&self.first, &self.second) + } + + pub fn step_by_step(&self) -> Vec { + let mut progression = vec![self.first.to_owned()]; + + for action in &self.actions { + let mut text = progression.last().unwrap().to_owned(); + + match action { + Action::INSERT(pos, chr) => { text.insert(*pos as usize, *chr); }, + Action::DELETE(pos) => { text.remove(*pos as usize - 1); }, + Action::REPLACE(_, _) => unimplemented!() + }; + + progression.push(text); + } + + progression + } + + fn create_lcs(&self) -> Grid { + let mut grid = Grid::new( + self.first.len() as u32 + 1, + self.second.len() as u32 + 1, + Cell(0, Arrow::NONE) + ); + + self.first.chars().enumerate().for_each(|(i, _)| grid[(i as u32 + 1, 0u32)] = Cell(0, Arrow::LEFT)); + self.second.chars().enumerate().for_each(|(i, _)| grid[(0u32, i as u32 + 1)] = Cell(0, Arrow::UP)); + + grid[(0, 0)] = Cell(0, Arrow::NONE); + + for (i, x) in self.first.chars().enumerate().map(|(i, c)| (i as u32 + 1, c)) { + for (j, y) in self.second.chars().enumerate().map(|(i, c)| (i as u32 + 1, c)) { + + if x == y { + grid[(i, j)] = Cell(grid[(i - 1, j - 1)].0 + 1, Arrow::DIAGONAL); + } else { + let left = grid[(i - 1, j)].0; + let up = grid[(i, j - 1)].0; + + grid[(i, j)] = match up >= left { + true => Cell(up, Arrow::UP), + false => Cell(left, Arrow::LEFT) + }; + } + } + } + + grid + } +} + +impl Display for Diff { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + let mut chars = self.first.chars().map(|c| c.to_string().normal()).collect::>(); + + for action in &self.actions { + match action { + Action::INSERT(pos, chr) => { chars.insert(*pos as usize, chr.to_string().green()) }, + Action::DELETE(pos) => { chars[*pos as usize - 1] = chars[*pos as usize - 1].to_owned().red().strikethrough() }, + Action::REPLACE(_, _) => unimplemented!() + }; + } + + for chr in chars { + write!(f, "{chr}")?; + } + + Ok(()) } } \ No newline at end of file diff --git a/src/grid.rs b/src/grid.rs index 85ae1cf..684a027 100644 --- a/src/grid.rs +++ b/src/grid.rs @@ -1,33 +1,46 @@ -use std::ops; +use std::{ops, fmt::Display}; -pub struct Grid { +pub struct Grid { width: usize, height: usize, buf: Vec } -impl Grid { - pub fn new(width: u32, height: u32) -> Grid { +impl Grid { + pub fn new(width: u32, height: u32, default: T) -> Grid { Grid { width: width as usize, height: height as usize, - buf: vec![T::default(); width as usize * height as usize] + buf: vec![default; width as usize * height as usize] } } } -impl ops::Index<(u32, u32)> for Grid { - type Output = T; - - fn index(&self, index: (u32, u32)) -> &Self::Output { - &self.buf[index.0 as usize * self.width + index.1 as usize] +impl Display for Grid { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + for y in 0..self.height { + for x in 0..self.width { + write!(f, "{}", self[(x as u32, y as u32)])?; + } + writeln!(f, "")?; + } + + Ok(()) } } -impl ops::IndexMut<(u32, u32)> for Grid { +impl ops::Index<(u32, u32)> for Grid { + type Output = T; + + fn index(&self, index: (u32, u32)) -> &Self::Output { + &self.buf[index.1 as usize * self.width + index.0 as usize] + } +} + +impl ops::IndexMut<(u32, u32)> for Grid { fn index_mut(&mut self, index: (u32, u32)) -> &mut Self::Output { - &mut self.buf[index.0 as usize * self.width + index.1 as usize] + &mut self.buf[index.1 as usize * self.width + index.0 as usize] } } \ No newline at end of file diff --git a/src/main.rs b/src/main.rs index a76cce3..b1ddcb1 100644 --- a/src/main.rs +++ b/src/main.rs @@ -5,6 +5,7 @@ use clap::Parser; use clap_derive::Parser; use diff::Diff; + #[derive(Parser)] struct Args { first: String, @@ -21,4 +22,9 @@ fn main() { return; } }; + + let inputs = diff.get_input(); + println!("First: {}\nSecond: {}\n", inputs.0, inputs.1); + + println!("{diff}"); }