navi/src/cheat.rs

243 lines
7.4 KiB
Rust
Raw Normal View History

2020-03-04 21:01:23 +00:00
use crate::display;
use crate::filesystem;
use crate::option::Config;
use regex::Regex;
use std::collections::HashMap;
use std::fs;
use std::io::Write;
2020-03-14 08:09:09 +00:00
#[derive(Debug, PartialEq)]
2020-03-04 21:01:23 +00:00
pub struct SuggestionOpts {
pub header_lines: u8,
pub column: Option<u8>,
2020-03-11 14:30:25 +00:00
pub delimiter: Option<String>,
2020-03-14 12:33:07 +00:00
pub suggestion_type: SuggestionType,
2020-03-04 21:01:23 +00:00
}
2020-03-14 12:33:07 +00:00
#[derive(Clone, Copy, Debug, PartialEq)]
pub enum SuggestionType {
2020-03-14 12:41:28 +00:00
/// fzf will not print any suggestions.
Disabled,
2020-03-14 12:41:28 +00:00
/// fzf will only select one of the suggestions
SingleSelection,
2020-03-14 12:41:28 +00:00
/// fzf will select multiple ones of the suggestions
MultipleSelections,
2020-03-14 12:41:28 +00:00
/// fzf will select one of the suggestions or use the Query
SingleRecommendation,
2020-03-14 12:41:28 +00:00
/// initial snippet selection
SnippetSelection,
}
pub type Suggestion = (String, Option<SuggestionOpts>);
2020-03-04 21:01:23 +00:00
2020-03-12 22:04:47 +00:00
fn remove_quote(txt: &str) -> String {
txt.replace('"', "").replace('\'', "")
}
2020-03-04 21:01:23 +00:00
fn parse_opts(text: &str) -> SuggestionOpts {
let mut header_lines: u8 = 0;
let mut column: Option<u8> = None;
let mut multi = false;
2020-03-14 08:09:09 +00:00
let mut allow_extra = false;
2020-03-11 14:30:25 +00:00
let mut delimiter: Option<String> = None;
2020-03-04 21:01:23 +00:00
let mut parts = text.split(' ');
while let Some(p) = parts.next() {
match p {
"--multi" => multi = true,
2020-03-14 08:09:09 +00:00
"--allow-extra" => allow_extra = true,
2020-03-11 14:30:25 +00:00
"--header" | "--headers" | "--header-lines" => {
2020-03-12 22:04:47 +00:00
header_lines = remove_quote(parts.next().unwrap()).parse::<u8>().unwrap()
2020-03-04 21:01:23 +00:00
}
2020-03-12 22:04:47 +00:00
"--column" => column = Some(remove_quote(parts.next().unwrap()).parse::<u8>().unwrap()),
"--delimiter" => delimiter = Some(remove_quote(parts.next().unwrap()).to_string()),
2020-03-04 21:01:23 +00:00
_ => (),
}
}
2020-03-14 12:41:28 +00:00
SuggestionOpts {
2020-03-04 21:01:23 +00:00
header_lines,
column,
2020-03-11 14:30:25 +00:00
delimiter,
2020-03-14 08:09:09 +00:00
suggestion_type: match (multi, allow_extra) {
2020-03-14 12:33:07 +00:00
(true, _) => SuggestionType::MultipleSelections, // multi wins over allow-extra
(false, true) => SuggestionType::SingleRecommendation,
(false, false) => SuggestionType::SingleSelection,
2020-03-14 08:09:09 +00:00
},
2020-03-14 12:41:28 +00:00
}
2020-03-04 21:01:23 +00:00
}
fn parse_variable_line(line: &str) -> (&str, &str, Option<SuggestionOpts>) {
let re = Regex::new(r"^\$\s*([^:]+):(.*)").unwrap();
let caps = re.captures(line).unwrap();
let variable = caps.get(1).unwrap().as_str().trim();
let mut command_plus_opts = caps.get(2).unwrap().as_str().split("---");
2020-03-14 15:46:33 +00:00
let command = command_plus_opts.next().unwrap();
let command_options = command_plus_opts.next().map(parse_opts);
2020-03-14 08:09:09 +00:00
(variable, command, command_options)
2020-03-04 21:01:23 +00:00
}
fn write_cmd(
tags: &str,
comment: &str,
snippet: &str,
tag_width: usize,
comment_width: usize,
stdin: &mut std::process::ChildStdin,
) -> bool {
if snippet.is_empty() {
true
} else {
stdin
.write_all(
display::format_line(
&tags[..],
&comment[..],
&snippet[3..],
tag_width,
comment_width,
)
.as_bytes(),
)
.is_ok()
}
}
2020-03-04 21:01:23 +00:00
fn read_file(
path: &str,
variables: &mut HashMap<String, Suggestion>,
2020-03-04 21:01:23 +00:00
stdin: &mut std::process::ChildStdin,
) {
let mut tags = String::from("");
let mut comment = String::from("");
let mut snippet = String::from("");
2020-03-07 21:03:51 +00:00
let (tag_width, comment_width) = *display::WIDTHS;
2020-03-04 21:01:23 +00:00
if let Ok(lines) = filesystem::read_lines(path) {
for l in lines {
let line = l.unwrap();
2020-03-14 21:14:30 +00:00
// blank
if line.is_empty() {
}
2020-03-04 21:01:23 +00:00
// tag
2020-03-14 21:14:30 +00:00
else if line.starts_with('%') {
if !write_cmd(&tags, &comment, &snippet, tag_width, comment_width, stdin) {
break;
}
snippet = String::from("");
2020-03-04 21:01:23 +00:00
tags = String::from(&line[2..]);
}
// metacomment
else if line.starts_with(';') {
}
2020-03-04 21:01:23 +00:00
// comment
else if line.starts_with('#') {
if !write_cmd(&tags, &comment, &snippet, tag_width, comment_width, stdin) {
break;
}
snippet = String::from("");
2020-03-04 21:01:23 +00:00
comment = String::from(&line[2..]);
}
// variable
else if line.starts_with('$') {
if !write_cmd(&tags, &comment, &snippet, tag_width, comment_width, stdin) {
break;
}
snippet = String::from("");
2020-03-14 08:09:09 +00:00
let (variable, command, opts) = parse_variable_line(&line);
2020-03-04 21:01:23 +00:00
variables.insert(
format!("{};{}", tags, variable),
(String::from(command), opts),
);
}
// snippet
else {
snippet.push_str(display::LINE_SEPARATOR);
snippet.push_str(&line);
2020-03-04 21:01:23 +00:00
}
}
}
write_cmd(&tags, &comment, &snippet, tag_width, comment_width, stdin);
2020-03-04 21:01:23 +00:00
}
2020-03-14 12:33:07 +00:00
pub fn read_all(
config: &Config,
stdin: &mut std::process::ChildStdin,
) -> HashMap<String, Suggestion> {
let mut variables: HashMap<String, Suggestion> = HashMap::new();
2020-03-04 21:01:23 +00:00
2020-03-12 22:04:47 +00:00
let mut fallback: String = String::from("");
let folders_str = config.path.as_ref().unwrap_or_else(|| {
if let Some(f) = filesystem::cheat_pathbuf() {
fallback = filesystem::pathbuf_to_string(f);
}
2020-03-12 22:04:47 +00:00
&fallback
});
2020-03-04 21:01:23 +00:00
let folders = folders_str.split(':');
for folder in folders {
if let Ok(paths) = fs::read_dir(folder) {
for path in paths {
let path_os_str = path.unwrap().path().into_os_string();
let path_str = path_os_str.to_str().unwrap();
if path_str.ends_with(".cheat") {
read_file(path_str, &mut variables, stdin);
}
2020-03-04 21:01:23 +00:00
}
}
}
variables
}
2020-03-14 08:09:09 +00:00
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_parse_variable_line() {
2020-03-14 12:33:07 +00:00
let (variable, command, command_options) =
parse_variable_line("$ user : echo -e \"$(whoami)\\nroot\" --- --allow-extra");
2020-03-14 08:09:09 +00:00
assert_eq!(command, " echo -e \"$(whoami)\\nroot\" ");
assert_eq!(variable, "user");
2020-03-14 12:33:07 +00:00
assert_eq!(
command_options,
Some(SuggestionOpts {
header_lines: 0,
column: None,
delimiter: None,
suggestion_type: SuggestionType::SingleRecommendation
})
);
2020-03-14 08:09:09 +00:00
}
use std::process::{Command, Stdio};
#[test]
fn test_read_file() {
let path = "tests/cheats/ssh.cheat";
let mut variables: HashMap<String, Suggestion> = HashMap::new();
2020-03-14 12:33:07 +00:00
let mut child = Command::new("cat").stdin(Stdio::piped()).spawn().unwrap();
2020-03-14 08:09:09 +00:00
let child_stdin = child.stdin.as_mut().unwrap();
2020-03-14 12:33:07 +00:00
read_file(path, &mut variables, child_stdin);
let mut result: HashMap<String, (String, std::option::Option<_>)> = HashMap::new();
result.insert(
"ssh;user".to_string(),
(
2020-03-14 15:46:33 +00:00
r#" echo -e "$(whoami)\nroot" "#.to_string(),
2020-03-14 12:33:07 +00:00
Some(SuggestionOpts {
header_lines: 0,
column: None,
delimiter: None,
suggestion_type: SuggestionType::SingleRecommendation,
}),
),
2020-03-14 08:09:09 +00:00
);
assert_eq!(variables, result);
}
}