use crate::edit::Edit;
use crate::replacer::Replacer;
use std::str;
pub(crate) struct Patcher<'a> {
edits: Vec<Edit>,
replacer: Option<&'a Replacer>,
}
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("Invalid line number {0}")]
LineNumber(u32),
#[error("invalid UTF-8 in replacement: {0}")]
Utf8(#[from] std::str::Utf8Error),
}
impl<'a> Patcher<'a> {
pub(crate) fn new(edits: Vec<Edit>, replacer: Option<&'a Replacer>) -> Self {
Self { edits, replacer }
}
pub(crate) fn patch(&self, mut lines: Vec<String>, delete: bool) -> Result<String, Error> {
if delete {
let mut indexes: Vec<u32> = self.edits.iter().map(|e| e.line_number).collect();
indexes.sort_unstable();
indexes.dedup();
indexes.reverse();
for index in indexes {
if index == 0 || index as usize > lines.len() {
return Err(Error::LineNumber(index));
}
let index_size = index as usize - 1;
lines.remove(index_size);
}
return Ok(lines.join("\n"));
}
for edit in &self.edits {
if edit.line_number == 0 || edit.line_number as usize > lines.len() {
return Err(Error::LineNumber(edit.line_number));
}
let index = edit.line_number as usize - 1;
if let Some(replacer) = &self.replacer {
let replaced = &replacer.replace(edit.text.as_bytes());
let text = str::from_utf8(replaced)?;
lines[index] = text.to_string();
} else {
lines[index] = edit.text.clone();
}
}
Ok(lines.join("\n"))
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::path::PathBuf;
#[test]
fn patch_bad_number() {
let patcher = Patcher::new(
vec![
Edit {
file: PathBuf::from("f"),
line_number: 1,
text: "foo".to_string(),
},
Edit {
file: PathBuf::from("f"),
line_number: 3,
text: "bar".to_string(),
},
],
None,
);
let lines = vec!["a".to_string(), "b".to_string()];
let result = patcher.patch(lines, false);
assert!(matches!(result, Err(Error::LineNumber(3))));
}
#[test]
fn patch() {
let patcher = Patcher::new(
vec![
Edit {
file: PathBuf::from("f"),
line_number: 2,
text: "foo".to_string(),
},
Edit {
file: PathBuf::from("f"),
line_number: 3,
text: "bar".to_string(),
},
],
None,
);
let lines = vec!["a".to_string(), "b".to_string(), "c".to_string()];
let result = patcher.patch(lines, false);
assert!(result.is_ok());
assert_eq!(result.unwrap(), "a\nfoo\nbar");
}
}