pay-respects/src/corrections.rs

119 lines
3.4 KiB
Rust
Raw Normal View History

2023-07-30 18:40:18 +02:00
use std::collections::HashMap;
use rule_parser::parse_rules;
2023-07-31 03:21:06 +02:00
use crate::shell::{command_output, PRIVILEGE_LIST};
2023-07-30 18:40:18 +02:00
use crate::style::highlight_difference;
2023-07-30 23:48:04 +02:00
pub fn correct_command(shell: &str, last_command: &str) -> Option<String> {
2023-07-31 00:27:36 +02:00
let command_output = command_output(shell, last_command);
2023-07-30 18:40:18 +02:00
let split_command = last_command.split_whitespace().collect::<Vec<&str>>();
2023-07-31 03:21:06 +02:00
let command = match PRIVILEGE_LIST.contains(&split_command[0]) {
true => split_command.get(1).expect("No command found."),
false => split_command.first().expect("No command found."),
2023-07-30 18:40:18 +02:00
};
2023-07-31 03:21:06 +02:00
if !PRIVILEGE_LIST.contains(command) {
let suggest = match_pattern("privilege", &command_output);
2023-07-30 18:40:18 +02:00
if let Some(suggest) = suggest {
2023-07-31 00:27:36 +02:00
let suggest = eval_suggest(&suggest, last_command);
2023-07-30 23:48:04 +02:00
return Some(suggest);
2023-07-30 18:40:18 +02:00
}
}
let suggest = match_pattern(command, &command_output);
if let Some(suggest) = suggest {
2023-07-31 00:27:36 +02:00
let suggest = eval_suggest(&suggest, last_command);
2023-07-31 03:21:06 +02:00
if PRIVILEGE_LIST.contains(command) {
return Some(format!("{} {}", split_command[0], suggest));
2023-07-30 23:48:04 +02:00
}
return Some(suggest);
2023-07-30 18:40:18 +02:00
}
None
}
fn match_pattern(command: &str, error_msg: &str) -> Option<String> {
let rules = parse_rules!("rules");
if rules.contains_key(command) {
let suggest = rules.get(command).unwrap();
for (pattern, suggest) in suggest {
for pattern in pattern {
if error_msg.contains(pattern) {
return Some(suggest.to_owned().to_string());
}
}
}
None
} else {
None
}
}
fn eval_suggest(suggest: &str, last_command: &str) -> String {
let mut suggest = suggest.to_owned();
if suggest.contains("{{command}}") {
suggest = suggest.replace("{{command}}", last_command);
}
while suggest.contains("{{command") {
let placeholder_start = "{{command";
let placeholder_end = "}}";
let placeholder = suggest.find(placeholder_start).unwrap()
..suggest.find(placeholder_end).unwrap() + placeholder_end.len();
let range = suggest[placeholder.to_owned()].trim_matches(|c| c == '[' || c == ']');
if let Some((start, end)) = range.split_once(':') {
2023-07-30 20:14:04 +02:00
let start = match start {
"" => 0,
_ => start.parse::<usize>().unwrap(),
};
let end = match end {
"" => last_command.split_whitespace().count(),
_ => end.parse::<usize>().unwrap(),
};
2023-07-30 18:40:18 +02:00
let split_command = last_command.split_whitespace().collect::<Vec<&str>>();
let command = split_command[start..end].join(" ");
suggest = suggest.replace(&suggest[placeholder], &command);
} else {
let range = range.parse::<usize>().unwrap();
let split_command = last_command.split_whitespace().collect::<Vec<&str>>();
let command = split_command[range].to_owned();
suggest = suggest.replace(&suggest[placeholder], &command);
}
}
suggest
}
2023-07-30 23:48:04 +02:00
pub fn confirm_correction(shell: &str, command: &str, last_command: &str) {
2023-07-31 00:27:36 +02:00
println!(
"Did you mean {}?",
highlight_difference(command, last_command)
);
2023-07-30 18:40:18 +02:00
println!("Press enter to execute the corrected command. Or press Ctrl+C to exit.");
std::io::stdin().read_line(&mut String::new()).unwrap();
2023-07-30 23:48:04 +02:00
2023-07-31 03:21:06 +02:00
for p in PRIVILEGE_LIST {
if command.starts_with(p) {
2023-07-31 01:58:37 +02:00
let command = command.replace(p, "");
2023-07-31 03:21:06 +02:00
println!("{} {}", p, command);
std::process::Command::new(p.trim())
2023-07-31 01:58:37 +02:00
.arg(shell)
.arg("-c")
.arg(command)
.spawn()
.expect("failed to execute process")
.wait()
.expect("failed to wait on process");
return;
}
2023-07-31 00:27:36 +02:00
}
2023-07-31 01:58:37 +02:00
std::process::Command::new(shell)
.arg("-c")
.arg(command)
.spawn()
.expect("failed to execute process")
.wait()
.expect("failed to wait on process");
2023-07-30 18:40:18 +02:00
}