use std::collections::HashSet;
use std::env;
use std::fs::read_dir;
use std::io::Write;
use std::iter::FromIterator;
use std::os::unix::fs::PermissionsExt;
use std::path::{is_separator, MAIN_SEPARATOR};
use std::sync::Arc;
use linefeed::complete::{Completer, Completion, Suffix};
use linefeed::terminal::Terminal;
use linefeed::Prompter;
use crate::parsers;
use crate::shell;
use crate::tools;
pub struct BinCompleter {
pub sh: Arc<shell::Shell>,
}
pub struct CdCompleter;
pub struct PathCompleter;
impl<Term: Terminal> Completer<Term> for BinCompleter {
fn complete(
&self,
word: &str,
_reader: &Prompter<Term>,
_start: usize,
_end: usize,
) -> Option<Vec<Completion>> {
let sh = Arc::try_unwrap(self.sh.clone());
match sh {
Ok(x) => Some(complete_bin(&x, word)),
Err(e) => Some(complete_bin(&e, word)),
}
}
}
impl<Term: Terminal> Completer<Term> for PathCompleter {
fn complete(
&self,
word: &str,
_reader: &Prompter<Term>,
_start: usize,
_end: usize,
) -> Option<Vec<Completion>> {
Some(complete_path(word, false))
}
}
impl<Term: Terminal> Completer<Term> for CdCompleter {
fn complete(
&self,
word: &str,
_reader: &Prompter<Term>,
_start: usize,
_end: usize,
) -> Option<Vec<Completion>> {
Some(complete_path(word, true))
}
}
pub fn complete_path(word: &str, for_dir: bool) -> Vec<Completion> {
let mut res = Vec::new();
let tokens = parsers::parser_line::cmd_to_tokens(word);
let (path, path_sep) = if tokens.is_empty() {
(String::new(), String::new())
} else {
let (ref _path_sep, ref _path) = tokens[tokens.len() - 1];
(_path.clone(), _path_sep.clone())
};
let (_dir_orig, _) = split_path(&path);
let dir_orig = if let Some(_dir) = _dir_orig { _dir } else { "" };
let mut path_extended = path.clone();
if shell::needs_expand_home(&path_extended) {
shell::expand_home_string(&mut path_extended)
}
let (_dir_lookup, file_name) = split_path(&path_extended);
let dir_lookup = _dir_lookup.unwrap_or(".");
if let Ok(entries) = read_dir(dir_lookup) {
for entry in entries {
if let Ok(entry) = entry {
let pathbuf = entry.path();
let is_dir = pathbuf.is_dir();
if for_dir && !is_dir {
continue;
}
let entry_name = entry.file_name();
if let Ok(_path) = entry_name.into_string() {
if _path.starts_with(file_name) {
let (name, display) = if dir_orig != "" {
(
format!("{}{}{}", dir_orig, MAIN_SEPARATOR, _path),
Some(_path),
)
} else {
(_path, None)
};
let mut name = str::replace(name.as_str(), "//", "/");
if path_sep.is_empty() {
name = tools::escape_path(&name);
}
let mut quoted = false;
if !path_sep.is_empty() {
name = tools::wrap_sep_string(&path_sep, &name);
quoted = true;
}
let suffix = if is_dir {
if quoted {
name.pop();
}
Suffix::Some(MAIN_SEPARATOR)
} else {
Suffix::Default
};
res.push(Completion {
completion: name,
display,
suffix,
});
}
}
}
}
}
res
}
fn split_path(path: &str) -> (Option<&str>, &str) {
match path.rfind(is_separator) {
Some(pos) => (Some(&path[..=pos]), &path[pos + 1..]),
None => (None, path),
}
}
fn complete_bin(sh: &shell::Shell, path: &str) -> Vec<Completion> {
let mut res = Vec::new();
let (_, fname) = split_path(path);
let env_path;
match env::var("PATH") {
Ok(x) => env_path = x,
Err(e) => {
println_stderr!("cicada: env error when complete_bin: {:?}", e);
return res;
}
}
let mut checker: HashSet<String> = HashSet::new();
for alias in sh.alias.keys() {
if !alias.starts_with(fname) {
continue;
}
if checker.contains(alias) {
continue;
}
checker.insert(alias.clone());
res.push(Completion {
completion: alias.to_owned(),
display: None,
suffix: Suffix::Default,
});
}
let builtins = vec![
"alias", "bg", "cd", "cinfo", "exec", "exit", "export", "fg", "history", "jobs", "source",
"unalias", "vox",
];
for item in &builtins {
if !item.starts_with(fname) {
continue;
}
if checker.contains(item.clone()) {
continue;
}
checker.insert(item.to_string());
res.push(Completion {
completion: item.to_string(),
display: None,
suffix: Suffix::Default,
});
}
let vec_path: Vec<&str> = env_path.split(':').collect();
let path_list: HashSet<&str> = HashSet::from_iter(vec_path.iter().cloned());
for p in &path_list {
if let Ok(list) = read_dir(p) {
for entry in list {
if let Ok(entry) = entry {
if let Ok(name) = entry.file_name().into_string() {
if name.starts_with(fname) {
let _mode;
match entry.metadata() {
Ok(x) => _mode = x,
Err(e) => {
println_stderr!("cicada: metadata error: {:?}", e);
continue;
}
}
let mode = _mode.permissions().mode();
if mode & 0o111 == 0 {
continue;
}
if checker.contains(&name) {
continue;
}
let display = None;
let suffix = Suffix::Default;
checker.insert(name.clone());
let name_e = tools::escape_path(&name);
res.push(Completion {
completion: name_e,
display,
suffix,
});
}
}
}
}
}
}
res
}
#[cfg(test)]
mod tests {
use super::split_path;
#[test]
fn test_split_path() {
assert_eq!(split_path(""), (None, ""));
assert_eq!(split_path(""), (None, ""));
}
}