mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-09 11:38:49 +00:00
77aeb6a2a8
Drop support for history file version 1. ParseExecutionContext no longer contains an OperationContext because in my first implementation, ParseExecutionContext didn't have interior mutability. We should probably try to add it back. Add a few to-do style comments. Search for "todo!" and "PORTING". Co-authored-by: Xiretza <xiretza@xiretza.xyz> (complete, wildcard, expand, history, history/file) Co-authored-by: Henrik Hørlück Berg <36937807+henrikhorluck@users.noreply.github.com> (builtins/set)
2723 lines
98 KiB
Rust
2723 lines
98 KiB
Rust
use std::{
|
|
cmp::Ordering,
|
|
collections::{BTreeMap, HashMap, HashSet},
|
|
mem,
|
|
pin::Pin,
|
|
sync::{
|
|
atomic::{self, AtomicUsize},
|
|
Mutex,
|
|
},
|
|
time::{Duration, Instant},
|
|
};
|
|
|
|
use crate::{
|
|
common::charptr2wcstring,
|
|
util::wcsfilecmp,
|
|
wchar_ffi::{AsWstr, WCharFromFFI, WCharToFFI},
|
|
};
|
|
use bitflags::bitflags;
|
|
use cxx::{CxxWString, UniquePtr};
|
|
use once_cell::sync::Lazy;
|
|
use printf_compat::sprintf;
|
|
use widestring::U32CString;
|
|
|
|
use crate::{
|
|
abbrs::with_abbrs,
|
|
autoload::Autoload,
|
|
builtins::shared::{builtin_exists, builtin_get_desc, builtin_get_names},
|
|
common::{
|
|
escape, unescape_string, valid_var_name_char, ScopeGuard, UnescapeFlags,
|
|
UnescapeStringStyle,
|
|
},
|
|
env::{EnvMode, EnvStack, Environment},
|
|
exec::exec_subshell,
|
|
expand::{
|
|
expand_escape_string, expand_escape_variable, expand_one, expand_string,
|
|
expand_to_receiver, ExpandFlags, ExpandResultCode,
|
|
},
|
|
flog::{FLOG, FLOGF},
|
|
function,
|
|
history::{history_session_id, History},
|
|
operation_context::OperationContext,
|
|
parse_constants::SourceRange,
|
|
parse_util::{
|
|
parse_util_cmdsubst_extent, parse_util_process_extent, parse_util_unescape_wildcards,
|
|
},
|
|
parser::{Block, Parser},
|
|
parser_keywords::parser_keywords_is_subcommand,
|
|
path::{path_get_path, path_try_get_path},
|
|
tokenizer::{variable_assignment_equals_pos, Tok, TokFlags, TokenType, Tokenizer},
|
|
wchar::{wstr, WString, L},
|
|
wchar_ext::WExt,
|
|
wcstringutil::{
|
|
string_fuzzy_match_string, string_prefixes_string, string_prefixes_string_case_insensitive,
|
|
StringFuzzyMatch,
|
|
},
|
|
wildcard::{wildcard_complete, wildcard_has, wildcard_match},
|
|
wutil::{gettext::wgettext_impl_do_not_use_directly, wgettext, wrealpath},
|
|
};
|
|
|
|
// Completion description strings, mostly for different types of files, such as sockets, block
|
|
// devices, etc.
|
|
//
|
|
// There are a few more completion description strings defined in expand.rs. Maybe all completion
|
|
// description strings should be defined in the same file?
|
|
|
|
/// Description for ~USER completion.
|
|
static COMPLETE_USER_DESC: Lazy<&wstr> = Lazy::new(|| wgettext!("Home for %ls"));
|
|
|
|
/// Description for short variables. The value is concatenated to this description.
|
|
static COMPLETE_VAR_DESC_VAL: Lazy<&wstr> = Lazy::new(|| wgettext!("Variable: %ls"));
|
|
|
|
/// Description for abbreviations.
|
|
static ABBR_DESC: Lazy<&wstr> = Lazy::new(|| wgettext!("Abbreviation: %ls"));
|
|
|
|
/// The special cased translation macro for completions. The empty string needs to be special cased,
|
|
/// since it can occur, and should not be translated. (Gettext returns the version information as
|
|
/// the response).
|
|
#[allow(non_snake_case)]
|
|
fn C_(s: &wstr) -> &'static wstr {
|
|
if s.is_empty() {
|
|
L!("")
|
|
} else {
|
|
wgettext_impl_do_not_use_directly(
|
|
U32CString::from_ustr(s)
|
|
.expect("translation string without NUL bytes")
|
|
.as_slice_with_nul(),
|
|
)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, Default, PartialEq, Eq, Debug)]
|
|
pub struct CompletionMode {
|
|
/// If set, skip file completions.
|
|
pub no_files: bool,
|
|
pub force_files: bool,
|
|
|
|
/// If set, require a parameter after completion.
|
|
pub requires_param: bool,
|
|
}
|
|
|
|
/// Character that separates the completion and description on programmable completions.
|
|
pub const PROG_COMPLETE_SEP: char = '\t';
|
|
|
|
bitflags! {
|
|
#[derive(Copy, Clone, Debug, Default, PartialEq, Eq)]
|
|
pub struct CompleteFlags: u8 {
|
|
/// Do not insert space afterwards if this is the only completion. (The default is to try insert
|
|
/// a space).
|
|
const NO_SPACE = 1 << 0;
|
|
/// This is not the suffix of a token, but replaces it entirely.
|
|
const REPLACES_TOKEN = 1 << 1;
|
|
/// This completion may or may not want a space at the end - guess by checking the last
|
|
/// character of the completion.
|
|
const AUTO_SPACE = 1 << 2;
|
|
/// This completion should be inserted as-is, without escaping.
|
|
const DONT_ESCAPE = 1 << 3;
|
|
/// If you do escape, don't escape tildes.
|
|
const DONT_ESCAPE_TILDES = 1 << 4;
|
|
/// Do not sort supplied completions
|
|
const DONT_SORT = 1 << 5;
|
|
/// This completion looks to have the same string as an existing argument.
|
|
const DUPLICATES_ARGUMENT = 1 << 6;
|
|
/// This completes not just a token but replaces the entire commandline.
|
|
const REPLACES_COMMANDLINE = 1 << 7;
|
|
}
|
|
}
|
|
|
|
/// Function which accepts a completion string and returns its description.
|
|
pub type DescriptionFunc = Box<dyn Fn(&wstr) -> WString>;
|
|
|
|
/// Helper to return a [`DescriptionFunc`] for a constant string.
|
|
pub fn const_desc(s: &wstr) -> DescriptionFunc {
|
|
let s = s.to_owned();
|
|
Box::new(move |_| s.clone())
|
|
}
|
|
|
|
pub type CompletionList = Vec<Completion>;
|
|
|
|
/// This is an individual completion entry, i.e. the result of an expansion of a completion rule.
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
pub struct Completion {
|
|
/// The completion string.
|
|
pub completion: WString,
|
|
/// The description for this completion.
|
|
pub description: WString,
|
|
/// The type of fuzzy match.
|
|
pub r#match: StringFuzzyMatch,
|
|
/// Flags determining the completion behavior.
|
|
pub flags: CompleteFlags,
|
|
}
|
|
|
|
impl Default for Completion {
|
|
fn default() -> Self {
|
|
Self {
|
|
completion: Default::default(),
|
|
description: Default::default(),
|
|
r#match: StringFuzzyMatch::exact_match(),
|
|
flags: Default::default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<WString> for Completion {
|
|
fn from(completion: WString) -> Completion {
|
|
Completion {
|
|
completion,
|
|
..Default::default()
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Completion {
|
|
pub fn new(
|
|
completion: WString,
|
|
description: WString,
|
|
r#match: StringFuzzyMatch,
|
|
flags: CompleteFlags,
|
|
) -> Self {
|
|
let flags = resolve_auto_space(&completion, flags);
|
|
Self {
|
|
completion,
|
|
description,
|
|
r#match,
|
|
flags,
|
|
}
|
|
}
|
|
|
|
pub fn from_completion(completion: WString) -> Self {
|
|
Self::with_desc(completion, WString::new())
|
|
}
|
|
|
|
pub fn with_desc(completion: WString, description: WString) -> Self {
|
|
Self::new(
|
|
completion,
|
|
description,
|
|
StringFuzzyMatch::exact_match(),
|
|
CompleteFlags::empty(),
|
|
)
|
|
}
|
|
|
|
/// Returns whether this replaces its token.
|
|
pub fn replaces_token(&self) -> bool {
|
|
self.flags.contains(CompleteFlags::REPLACES_TOKEN)
|
|
}
|
|
|
|
/// Returns whether this replaces the entire commandline.
|
|
pub fn replaces_commandline(&self) -> bool {
|
|
self.flags.contains(CompleteFlags::REPLACES_COMMANDLINE)
|
|
}
|
|
|
|
/// Returns the completion's match rank. Lower ranks are better completions.
|
|
pub fn rank(&self) -> u32 {
|
|
self.r#match.rank()
|
|
}
|
|
|
|
/// If this completion replaces the entire token, prepend a prefix. Otherwise do nothing.
|
|
pub fn prepend_token_prefix(&mut self, prefix: &wstr) {
|
|
if self.flags.contains(CompleteFlags::REPLACES_TOKEN) {
|
|
self.completion.insert_utfstr(0, prefix)
|
|
}
|
|
}
|
|
}
|
|
|
|
impl CompletionRequestOptions {
|
|
/// Options for an autosuggestion.
|
|
pub fn autosuggest() -> Self {
|
|
Self {
|
|
autosuggestion: true,
|
|
descriptions: false,
|
|
fuzzy_match: false,
|
|
}
|
|
}
|
|
|
|
/// Options for a "normal" completion.
|
|
pub fn normal() -> Self {
|
|
Self {
|
|
autosuggestion: false,
|
|
descriptions: true,
|
|
fuzzy_match: true,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// A completion receiver accepts completions. It is essentially a wrapper around `Vec` with
|
|
/// some conveniences.
|
|
pub struct CompletionReceiver {
|
|
/// Our list of completions.
|
|
completions: Vec<Completion>,
|
|
/// The maximum number of completions to add. If our list length exceeds this, then new
|
|
/// completions are not added. Note 0 has no special significance here - use
|
|
/// `usize::MAX` instead.
|
|
limit: usize,
|
|
}
|
|
|
|
// We are only wrapping a `Vec<Completion>`, any non-mutable methods can be safely deferred to the
|
|
// Vec-impl
|
|
impl std::ops::Deref for CompletionReceiver {
|
|
type Target = [Completion];
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
self.completions.as_slice()
|
|
}
|
|
}
|
|
|
|
impl std::ops::DerefMut for CompletionReceiver {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
self.completions.as_mut_slice()
|
|
}
|
|
}
|
|
|
|
impl CompletionReceiver {
|
|
/// Construct as empty, with a limit.
|
|
pub fn new(limit: usize) -> Self {
|
|
Self {
|
|
completions: vec![],
|
|
limit,
|
|
}
|
|
}
|
|
|
|
/// Acquire an existing list, with a limit.
|
|
pub fn from_list(completions: Vec<Completion>, limit: usize) -> Self {
|
|
Self { completions, limit }
|
|
}
|
|
|
|
/// Add a completion.
|
|
/// \return true on success, false if this would overflow the limit.
|
|
#[must_use]
|
|
pub fn add(&mut self, comp: impl Into<Completion>) -> bool {
|
|
if self.completions.len() >= self.limit {
|
|
return false;
|
|
}
|
|
self.completions.push(comp.into());
|
|
return true;
|
|
}
|
|
|
|
/// Adds a completion with the given string, and default other properties. Returns `true` on
|
|
/// success, `false` if this would overflow the limit.
|
|
#[must_use]
|
|
pub fn extend(
|
|
&mut self,
|
|
iter: impl IntoIterator<Item = Completion, IntoIter = impl ExactSizeIterator<Item = Completion>>,
|
|
) -> bool {
|
|
let iter = iter.into_iter();
|
|
if iter.len() > self.limit - self.completions.len() {
|
|
return false;
|
|
}
|
|
self.completions.extend(iter);
|
|
// this only fails if the ExactSizeIterator impl is bogus
|
|
assert!(
|
|
self.completions.len() <= self.limit,
|
|
"ExactSizeIterator returned more items than it should"
|
|
);
|
|
|
|
true
|
|
}
|
|
|
|
/// Clear the list of completions. This retains the storage inside `completions` which can be
|
|
/// useful to prevent allocations.
|
|
pub fn clear(&mut self) {
|
|
self.completions.clear();
|
|
}
|
|
|
|
/// Returns whether our completion list is empty.
|
|
pub fn empty(&self) -> bool {
|
|
self.completions.is_empty()
|
|
}
|
|
|
|
/// Returns how many completions we have stored.
|
|
pub fn size(&self) -> usize {
|
|
self.completions.len()
|
|
}
|
|
|
|
/// Returns the list of completions.
|
|
pub fn get_list(&self) -> &[Completion] {
|
|
&self.completions
|
|
}
|
|
|
|
/// Returns the list of completions.
|
|
pub fn get_list_mut(&mut self) -> &mut [Completion] {
|
|
&mut self.completions
|
|
}
|
|
|
|
/// Returns the list of completions, clearing it.
|
|
pub fn take(&mut self) -> Vec<Completion> {
|
|
std::mem::take(&mut self.completions)
|
|
}
|
|
|
|
/// Returns a new, empty receiver whose limit is our remaining capacity.
|
|
/// This is useful for e.g. recursive calls when you want to act on the result before adding it.
|
|
pub fn subreceiver(&self) -> Self {
|
|
let remaining_capacity = self
|
|
.limit
|
|
.checked_sub(self.completions.len())
|
|
.expect("length should never be larger than limit");
|
|
Self::new(remaining_capacity)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
|
pub enum CompleteOptionType {
|
|
/// no option
|
|
ArgsOnly,
|
|
/// `-x`
|
|
Short,
|
|
/// `-foo`
|
|
SingleLong,
|
|
/// `--foo`
|
|
DoubleLong,
|
|
}
|
|
|
|
/// Struct describing a completion rule for options to a command.
|
|
///
|
|
/// If option is empty, the comp field must not be empty and contains a list of arguments to the
|
|
/// command.
|
|
///
|
|
/// The type field determines how the option is to be interpreted: either empty (args_only) or
|
|
/// short, single-long ("old") or double-long ("GNU"). An invariant is that the option is empty if
|
|
/// and only if the type is args_only.
|
|
///
|
|
/// If option is non-empty, it specifies a switch for the command. If \c comp is also not empty, it
|
|
/// contains a list of non-switch arguments that may only follow directly after the specified
|
|
/// switch.
|
|
#[derive(Clone, Debug)]
|
|
struct CompleteEntryOpt {
|
|
/// Text of the option (like 'foo').
|
|
option: WString,
|
|
/// Arguments to the option; may be a subshell expression expanded at evaluation time.
|
|
comp: WString,
|
|
/// Description of the completion.
|
|
desc: WString,
|
|
/// Conditions under which to use the option, expanded and evaluated at completion time.
|
|
conditions: Vec<WString>,
|
|
/// Type of the option: `ArgsOnly`, `Short`, `SingleLong`, or `DoubleLong`.
|
|
typ: CompleteOptionType,
|
|
/// Determines how completions should be performed on the argument after the switch.
|
|
result_mode: CompletionMode,
|
|
/// Completion flags.
|
|
flags: CompleteFlags,
|
|
}
|
|
|
|
impl CompleteEntryOpt {
|
|
pub fn localized_desc(&self) -> &'static wstr {
|
|
C_(&self.desc)
|
|
}
|
|
|
|
pub fn expected_dash_count(&self) -> usize {
|
|
match self.typ {
|
|
CompleteOptionType::ArgsOnly => 0,
|
|
CompleteOptionType::Short | CompleteOptionType::SingleLong => 1,
|
|
CompleteOptionType::DoubleLong => 2,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Last value used in the order field of [`CompletionEntry`].
|
|
static complete_order: AtomicUsize = AtomicUsize::new(0);
|
|
|
|
struct CompletionEntry {
|
|
/// List of all options.
|
|
options: Vec<CompleteEntryOpt>,
|
|
/// Order for when this completion was created. This aids in outputting completions sorted by
|
|
/// time.
|
|
order: usize,
|
|
}
|
|
|
|
impl CompletionEntry {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
options: vec![],
|
|
order: complete_order.fetch_add(1, atomic::Ordering::Relaxed),
|
|
}
|
|
}
|
|
|
|
/// Getters for option list.
|
|
pub fn get_options(&self) -> &[CompleteEntryOpt] {
|
|
&self.options
|
|
}
|
|
|
|
/// Adds an option.
|
|
pub fn add_option(&mut self, opt: CompleteEntryOpt) {
|
|
self.options.push(opt)
|
|
}
|
|
|
|
/// Remove all completion options in the specified entry that match the specified short / long
|
|
/// option strings. Returns true if it is now empty and should be deleted, false if it's not
|
|
/// empty.
|
|
pub fn remove_option(&mut self, option: &wstr, typ: CompleteOptionType) -> bool {
|
|
self.options
|
|
.retain(|opt| opt.option != option || opt.typ != typ);
|
|
self.options.is_empty()
|
|
}
|
|
}
|
|
|
|
/// Set of all completion entries. Keyed by the command name, and whether it is a path.
|
|
#[derive(Clone, Debug, PartialOrd, Ord, PartialEq, Eq, Hash)]
|
|
struct CompletionEntryIndex {
|
|
name: WString,
|
|
is_path: bool,
|
|
}
|
|
type CompletionEntryMap = BTreeMap<CompletionEntryIndex, CompletionEntry>;
|
|
static COMPLETION_MAP: Mutex<CompletionEntryMap> = Mutex::new(BTreeMap::new());
|
|
|
|
/// Completion "wrapper" support. The map goes from wrapping-command to wrapped-command-list.
|
|
type WrapperMap = HashMap<WString, Vec<WString>>;
|
|
static wrapper_map: Lazy<Mutex<WrapperMap>> = Lazy::new(|| Mutex::new(HashMap::new()));
|
|
|
|
/// Clear the [`CompleteFlags::AUTO_SPACE`] flag, and set [`CompleteFlags::NO_SPACE`] appropriately
|
|
/// depending on the suffix of the string.
|
|
fn resolve_auto_space(comp: &wstr, mut flags: CompleteFlags) -> CompleteFlags {
|
|
if flags.contains(CompleteFlags::AUTO_SPACE) {
|
|
flags -= CompleteFlags::AUTO_SPACE;
|
|
if let Some('/' | '=' | '@' | ':' | '.' | ',' | '-') = comp.as_char_slice().last() {
|
|
flags |= CompleteFlags::NO_SPACE;
|
|
}
|
|
}
|
|
|
|
flags
|
|
}
|
|
|
|
// If these functions aren't force inlined, it is actually faster to call
|
|
// stable_sort twice rather than to iterate once performing all comparisons in one go!
|
|
|
|
#[inline(always)]
|
|
fn natural_compare_completions(a: &Completion, b: &Completion) -> Ordering {
|
|
if (a.flags & b.flags).contains(CompleteFlags::DONT_SORT) {
|
|
// Both completions are from a source with the --keep-order flag.
|
|
return Ordering::Equal;
|
|
}
|
|
wcsfilecmp(&a.completion, &b.completion)
|
|
}
|
|
|
|
#[inline(always)]
|
|
fn compare_completions_by_duplicate_arguments(a: &Completion, b: &Completion) -> Ordering {
|
|
let ad = a.flags.contains(CompleteFlags::DUPLICATES_ARGUMENT);
|
|
let bd = b.flags.contains(CompleteFlags::DUPLICATES_ARGUMENT);
|
|
|
|
ad.cmp(&bd)
|
|
}
|
|
|
|
#[inline(always)]
|
|
fn compare_completions_by_tilde(a: &Completion, b: &Completion) -> Ordering {
|
|
if a.completion.is_empty() || b.completion.is_empty() {
|
|
return Ordering::Equal;
|
|
}
|
|
|
|
let at = a.completion.ends_with('~');
|
|
let bt = b.completion.ends_with('~');
|
|
|
|
at.cmp(&bt)
|
|
}
|
|
|
|
/// Unique the list of completions, without perturbing their order.
|
|
fn unique_completions_retaining_order(comps: &mut Vec<Completion>) {
|
|
let mut seen = HashSet::with_capacity(comps.len());
|
|
|
|
let pred = |c: &Completion| {
|
|
// Keep (return true) if insertion succeeds.
|
|
// todo!("don't clone here");
|
|
seen.insert(c.completion.to_owned())
|
|
};
|
|
|
|
comps.retain(pred);
|
|
}
|
|
|
|
/// Sorts and removes any duplicate completions in the completion list, then puts them in priority
|
|
/// order.
|
|
pub fn sort_and_prioritize(comps: &mut Vec<Completion>, flags: CompletionRequestOptions) {
|
|
if comps.is_empty() {
|
|
return;
|
|
}
|
|
|
|
// Find the best rank.
|
|
let best_rank = comps.iter().map(Completion::rank).min().unwrap();
|
|
|
|
// Throw out completions of worse ranks.
|
|
comps.retain(|c| c.rank() == best_rank);
|
|
|
|
// Deduplicate both sorted and unsorted results.
|
|
unique_completions_retaining_order(comps);
|
|
|
|
// Sort, provided DONT_SORT isn't set.
|
|
// Here we do not pass suppress_exact, so that exact matches appear first.
|
|
comps.sort_by(natural_compare_completions);
|
|
|
|
// Lastly, if this is for an autosuggestion, prefer to avoid completions that duplicate
|
|
// arguments, and penalize files that end in tilde - they're frequently autosave files from e.g.
|
|
// emacs. Also prefer samecase to smartcase.
|
|
if flags.autosuggestion {
|
|
comps.sort_by(|a, b| {
|
|
a.r#match
|
|
.case_fold
|
|
.cmp(&b.r#match.case_fold)
|
|
.then_with(|| compare_completions_by_duplicate_arguments(a, b))
|
|
.then_with(|| compare_completions_by_tilde(a, b))
|
|
})
|
|
}
|
|
}
|
|
|
|
/// Bag of data to support expanding a command's arguments using custom completions, including
|
|
/// the wrap chain.
|
|
struct CustomArgData<'a> {
|
|
/// The unescaped argument before the argument which is being completed, or empty if none.
|
|
previous_argument: WString,
|
|
/// The unescaped argument which is being completed, or empty if none.
|
|
current_argument: WString,
|
|
/// Whether a -- has been encountered, which suppresses options.
|
|
had_ddash: bool,
|
|
/// Whether to perform file completions.
|
|
/// This is an "out" parameter of the wrap chain walk: if any wrapped command suppresses file
|
|
/// completions this gets set to false.
|
|
do_file: bool,
|
|
/// Depth in the wrap chain.
|
|
wrap_depth: usize,
|
|
/// The list of variable assignments: escaped strings of the form VAR=VAL.
|
|
/// This may be temporarily appended to as we explore the wrap chain.
|
|
/// When completing, variable assignments are really set in a local scope.
|
|
var_assignments: &'a mut Vec<WString>,
|
|
/// The set of wrapped commands which we have visited, and so should not be explored again.
|
|
visited_wrapped_commands: HashSet<WString>,
|
|
}
|
|
|
|
impl<'a> CustomArgData<'a> {
|
|
pub fn new(var_assignments: &'a mut Vec<WString>) -> Self {
|
|
Self {
|
|
previous_argument: WString::new(),
|
|
current_argument: WString::new(),
|
|
had_ddash: false,
|
|
do_file: true,
|
|
wrap_depth: 0,
|
|
var_assignments,
|
|
visited_wrapped_commands: HashSet::new(),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Class representing an attempt to compute completions.
|
|
struct Completer<'ctx> {
|
|
/// The operation context for this completion.
|
|
ctx: &'ctx OperationContext<'ctx>,
|
|
/// Flags associated with the completion request.
|
|
flags: CompletionRequestOptions,
|
|
/// The output completions.
|
|
completions: CompletionReceiver,
|
|
/// Commands which we would have tried to load, if we had a parser.
|
|
needs_load: Vec<WString>,
|
|
/// Table of completions conditions that have already been tested and the corresponding test
|
|
/// results.
|
|
condition_cache: HashMap<WString, bool>,
|
|
}
|
|
|
|
static completion_autoloader: Lazy<Mutex<Autoload>> =
|
|
Lazy::new(|| Mutex::new(Autoload::new(L!("fish_complete_path"))));
|
|
|
|
impl<'ctx> Completer<'ctx> {
|
|
pub fn new(ctx: &'ctx OperationContext<'ctx>, flags: CompletionRequestOptions) -> Self {
|
|
Self {
|
|
ctx,
|
|
flags,
|
|
completions: CompletionReceiver::new(ctx.expansion_limit),
|
|
needs_load: vec![],
|
|
condition_cache: HashMap::new(),
|
|
}
|
|
}
|
|
|
|
pub fn perform_for_commandline(&mut self, cmdline: WString) {
|
|
// Limit recursion, in case a user-defined completion has cycles, or the completion for "x"
|
|
// wraps "A=B x" (#3474, #7344). No need to do that when there is no parser: this happens only
|
|
// for autosuggestions where we don't evaluate command substitutions or variable assignments.
|
|
let _decrement = if let Some(parser) = self.ctx.maybe_parser() {
|
|
let level = &mut parser.libdata_mut().pods.complete_recursion_level;
|
|
if *level >= 24 {
|
|
FLOG!(
|
|
error,
|
|
wgettext!("completion reached maximum recursion depth, possible cycle?"),
|
|
);
|
|
return;
|
|
}
|
|
*level += 1;
|
|
|
|
Some(ScopeGuard::new((), |()| {
|
|
let level = &mut parser.libdata_mut().pods.complete_recursion_level;
|
|
*level -= 1;
|
|
}))
|
|
} else {
|
|
None
|
|
};
|
|
|
|
let cursor_pos = cmdline.len();
|
|
let is_autosuggest = self.flags.autosuggestion;
|
|
|
|
// Find the process to operate on. The cursor may be past it (#1261), so backtrack
|
|
// until we know we're no longer in a space. But the space may actually be part of the
|
|
// argument (#2477).
|
|
let mut position_in_statement = cursor_pos;
|
|
while position_in_statement > 0 && cmdline.char_at(position_in_statement - 1) == ' ' {
|
|
position_in_statement -= 1;
|
|
}
|
|
|
|
// Get all the arguments.
|
|
let mut tokens = Vec::new();
|
|
parse_util_process_extent(&cmdline, position_in_statement, Some(&mut tokens));
|
|
let actual_token_count = tokens.len();
|
|
|
|
// Hack: fix autosuggestion by removing prefixing "and"s #6249.
|
|
if is_autosuggest {
|
|
tokens.retain(|token| !parser_keywords_is_subcommand(token.get_source(&cmdline)));
|
|
}
|
|
|
|
// Consume variable assignments in tokens strictly before the cursor.
|
|
// This is a list of (escaped) strings of the form VAR=VAL.
|
|
// TODO: filter_drain
|
|
let mut var_assignments = Vec::new();
|
|
for tok in &tokens {
|
|
if tok.location_in_or_at_end_of_source_range(cursor_pos) {
|
|
break;
|
|
}
|
|
let tok_src = tok.get_source(&cmdline);
|
|
if variable_assignment_equals_pos(tok_src).is_none() {
|
|
break;
|
|
}
|
|
var_assignments.push(tok_src.to_owned());
|
|
}
|
|
tokens.drain(..var_assignments.len());
|
|
|
|
// Empty process (cursor is after one of ;, &, |, \n, &&, || modulo whitespace).
|
|
let [first_token, ..] = tokens.as_slice() else {
|
|
// Don't autosuggest anything based on the empty string (generalizes #1631).
|
|
if is_autosuggest {
|
|
return;
|
|
}
|
|
self.complete_cmd(WString::new());
|
|
self.complete_abbr(WString::new());
|
|
return;
|
|
};
|
|
|
|
let effective_cmdline = if tokens.len() == actual_token_count {
|
|
&cmdline
|
|
} else {
|
|
&cmdline[first_token.offset()..]
|
|
};
|
|
|
|
if tokens.last().unwrap().type_ == TokenType::comment {
|
|
return;
|
|
}
|
|
tokens.retain(|tok| tok.type_ != TokenType::comment);
|
|
assert!(!tokens.is_empty());
|
|
|
|
let cmd_tok = tokens.first().unwrap();
|
|
let cur_tok = tokens.last().unwrap();
|
|
|
|
// Since fish does not currently support redirect in command position, we return here.
|
|
if cmd_tok.type_ != TokenType::string {
|
|
return;
|
|
}
|
|
if cur_tok.type_ == TokenType::error {
|
|
return;
|
|
}
|
|
for tok in &tokens {
|
|
// If there was an error, it was in the last token.
|
|
assert!(matches!(tok.type_, TokenType::string | TokenType::redirect));
|
|
}
|
|
// If we are completing a variable name or a tilde expansion user name, we do that and
|
|
// return. No need for any other completions.
|
|
let current_token = cur_tok.get_source(&cmdline);
|
|
if cur_tok.location_in_or_at_end_of_source_range(cursor_pos)
|
|
&& (self.try_complete_variable(current_token) || self.try_complete_user(current_token))
|
|
{
|
|
return;
|
|
}
|
|
|
|
if cmd_tok.location_in_or_at_end_of_source_range(cursor_pos) {
|
|
let equals_sign_pos = variable_assignment_equals_pos(current_token);
|
|
if equals_sign_pos.is_some() {
|
|
self.complete_param_expand(
|
|
current_token,
|
|
true, /* do_file */
|
|
false, /* handle_as_special_cd */
|
|
);
|
|
return;
|
|
}
|
|
// Complete command filename.
|
|
let current_token = current_token.to_owned();
|
|
self.complete_cmd(current_token.clone());
|
|
self.complete_abbr(current_token);
|
|
return;
|
|
}
|
|
// See whether we are in an argument, in a redirection or in the whitespace in between.
|
|
let mut in_redirection = cur_tok.type_ == TokenType::redirect;
|
|
|
|
let mut had_ddash = false;
|
|
let mut current_argument = L!("");
|
|
let mut previous_argument = L!("");
|
|
if cur_tok.type_ == TokenType::string
|
|
&& cur_tok.location_in_or_at_end_of_source_range(position_in_statement)
|
|
{
|
|
// If the cursor is in whitespace, then the "current" argument is empty and the
|
|
// previous argument is the matching one. But if the cursor was in or at the end
|
|
// of the argument, then the current argument is the matching one, and the
|
|
// previous argument is the one before it.
|
|
let cursor_in_whitespace = !cur_tok.location_in_or_at_end_of_source_range(cursor_pos);
|
|
if cursor_in_whitespace {
|
|
previous_argument = current_token;
|
|
} else {
|
|
current_argument = current_token;
|
|
if tokens.len() >= 2 {
|
|
let prev_tok = &tokens[tokens.len() - 2];
|
|
if prev_tok.type_ == TokenType::string {
|
|
previous_argument = prev_tok.get_source(&cmdline);
|
|
}
|
|
in_redirection = prev_tok.type_ == TokenType::redirect;
|
|
}
|
|
}
|
|
|
|
// Check to see if we have a preceding double-dash.
|
|
for tok in &tokens[..tokens.len() - 1] {
|
|
if tok.get_source(&cmdline) == L!("--") {
|
|
had_ddash = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
let mut do_file = false;
|
|
let mut handle_as_special_cd = false;
|
|
if in_redirection {
|
|
do_file = true;
|
|
} else {
|
|
// Try completing as an argument.
|
|
let mut arg_data = CustomArgData::new(&mut var_assignments);
|
|
arg_data.had_ddash = had_ddash;
|
|
|
|
let bias = cmdline.len() - effective_cmdline.len();
|
|
let command_range = SourceRange::new(cmd_tok.offset() - bias, cmd_tok.length());
|
|
|
|
let mut exp_command = cmd_tok.get_source(&cmdline).to_owned();
|
|
let mut prev = None;
|
|
let mut cur = None;
|
|
if expand_command_token(self.ctx, &mut exp_command) {
|
|
prev = unescape_string(previous_argument, UnescapeStringStyle::default());
|
|
cur = unescape_string(
|
|
current_argument,
|
|
UnescapeStringStyle::Script(UnescapeFlags::INCOMPLETE),
|
|
);
|
|
}
|
|
if let (Some(prev), Some(cur)) = (prev, cur) {
|
|
arg_data.previous_argument = prev;
|
|
arg_data.current_argument = cur;
|
|
// Have to walk over the command and its entire wrap chain. If any command
|
|
// disables do_file, then they all do.
|
|
self.walk_wrap_chain(
|
|
&exp_command,
|
|
effective_cmdline,
|
|
command_range,
|
|
&mut arg_data,
|
|
);
|
|
do_file = arg_data.do_file;
|
|
|
|
// If we're autosuggesting, and the token is empty, don't do file suggestions.
|
|
if is_autosuggest && arg_data.current_argument.is_empty() {
|
|
do_file = false;
|
|
}
|
|
}
|
|
|
|
// Hack. If we're cd, handle it specially (issue #1059, others).
|
|
handle_as_special_cd =
|
|
exp_command == L!("cd") || arg_data.visited_wrapped_commands.contains(L!("cd"));
|
|
}
|
|
|
|
// Maybe apply variable assignments.
|
|
let _restore_vars =
|
|
self.apply_var_assignments(var_assignments.iter().map(|s| s.as_utfstr()));
|
|
if self.ctx.check_cancel() {
|
|
return;
|
|
}
|
|
|
|
// This function wants the unescaped string.
|
|
self.complete_param_expand(current_argument, do_file, handle_as_special_cd);
|
|
|
|
// Escape '[' in the argument before completing it.
|
|
self.escape_opening_brackets(current_argument);
|
|
|
|
// Lastly mark any completions that appear to already be present in arguments.
|
|
self.mark_completions_duplicating_arguments(&cmdline, current_token, tokens);
|
|
}
|
|
|
|
pub fn acquire_completions(&mut self) -> Vec<Completion> {
|
|
self.completions.take()
|
|
}
|
|
|
|
pub fn acquire_needs_load(&mut self) -> Vec<WString> {
|
|
mem::take(&mut self.needs_load)
|
|
}
|
|
|
|
/// Test if the specified script returns zero. The result is cached, so that if multiple completions
|
|
/// use the same condition, it needs only be evaluated once. condition_cache_clear must be called
|
|
/// after a completion run to make sure that there are no stale completions.
|
|
fn condition_test(&mut self, condition: &wstr) -> bool {
|
|
if condition.is_empty() {
|
|
return true;
|
|
}
|
|
let Some(parser) = self.ctx.maybe_parser() else {
|
|
return false;
|
|
};
|
|
|
|
let cached_entry = self.condition_cache.get(condition);
|
|
if let Some(&entry) = cached_entry {
|
|
// Use the old value.
|
|
entry
|
|
} else {
|
|
// Compute new value and reinsert it.
|
|
let test_res = exec_subshell(
|
|
condition, parser, None, false, /* don't apply exit status */
|
|
) == 0;
|
|
self.condition_cache.insert(condition.to_owned(), test_res);
|
|
test_res
|
|
}
|
|
}
|
|
|
|
fn conditions_test(&mut self, conditions: &[WString]) -> bool {
|
|
conditions.iter().all(|c| self.condition_test(c))
|
|
}
|
|
|
|
/// Copy any strings in `possible_comp` which have the specified prefix to the
|
|
/// completer's completion array. The prefix may contain wildcards. The output
|
|
/// will consist of [`Completion`] structs.
|
|
///
|
|
/// There are three ways to specify descriptions for each completion. Firstly,
|
|
/// if a description has already been added to the completion, it is _not_
|
|
/// replaced. Secondly, if the `desc_func` function is specified, use it to
|
|
/// determine a dynamic completion. Thirdly, if none of the above are available,
|
|
/// the `desc` string is used as a description.
|
|
///
|
|
/// - `wc_escaped`: the prefix, possibly containing wildcards. The wildcard should not have
|
|
/// been unescaped, i.e. '*' should be used for any string, not the `ANY_STRING` character.
|
|
/// - `desc_func`: the function that generates a description for those completions without an
|
|
/// embedded description
|
|
/// - `possible_comp`: the list of possible completions to iterate over
|
|
/// - `flags`: The flags controlling completion
|
|
/// - `extra_expand_flags`: Additional flags controlling expansion.
|
|
fn complete_strings(
|
|
&mut self,
|
|
wc_escaped: &wstr,
|
|
desc_func: &DescriptionFunc,
|
|
possible_comp: &[Completion],
|
|
flags: CompleteFlags,
|
|
extra_expand_flags: ExpandFlags,
|
|
) {
|
|
let mut tmp = wc_escaped.to_owned();
|
|
if !expand_one(
|
|
&mut tmp,
|
|
self.expand_flags()
|
|
| extra_expand_flags
|
|
| ExpandFlags::SKIP_CMDSUBST
|
|
| ExpandFlags::SKIP_WILDCARDS,
|
|
self.ctx,
|
|
None,
|
|
) {
|
|
return;
|
|
}
|
|
|
|
let wc = parse_util_unescape_wildcards(&tmp);
|
|
for comp in possible_comp {
|
|
let comp_str = &comp.completion;
|
|
if !comp_str.is_empty() {
|
|
let expand_flags = self.expand_flags() | extra_expand_flags;
|
|
wildcard_complete(
|
|
comp_str,
|
|
&wc,
|
|
Some(desc_func),
|
|
Some(&mut self.completions),
|
|
expand_flags,
|
|
flags,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn expand_flags(&self) -> ExpandFlags {
|
|
let mut result = ExpandFlags::empty();
|
|
result.set(ExpandFlags::SKIP_CMDSUBST, self.flags.autosuggestion);
|
|
result.set(ExpandFlags::FUZZY_MATCH, self.flags.fuzzy_match);
|
|
result.set(ExpandFlags::GEN_DESCRIPTIONS, self.flags.descriptions);
|
|
result
|
|
}
|
|
|
|
/// If command to complete is short enough, substitute the description with the whatis information
|
|
/// for the executable.
|
|
fn complete_cmd_desc(&mut self, s: &wstr) {
|
|
let Some(parser) = self.ctx.maybe_parser() else {
|
|
return;
|
|
};
|
|
|
|
let cmd = if let Some(pos) = s.chars().rposition(|c| c == '/') {
|
|
if pos + 1 > s.len() {
|
|
return;
|
|
}
|
|
&s[pos + 1..]
|
|
} else {
|
|
s
|
|
};
|
|
|
|
// Using apropos with a single-character search term produces far too many results - require at
|
|
// least two characters if we don't know the location of the whatis-database.
|
|
if cmd.len() < 2 {
|
|
return;
|
|
}
|
|
|
|
if wildcard_has(cmd) {
|
|
return;
|
|
}
|
|
|
|
let keep_going =
|
|
self.completions.get_list().iter().any(|c| {
|
|
c.completion.is_empty() || c.completion.as_char_slice().last() != Some(&'/')
|
|
});
|
|
if !keep_going {
|
|
return;
|
|
}
|
|
|
|
let lookup_cmd: WString = [L!("__fish_describe_command "), &escape(cmd)]
|
|
.into_iter()
|
|
.collect();
|
|
|
|
// First locate a list of possible descriptions using a single call to apropos or a direct
|
|
// search if we know the location of the whatis database. This can take some time on slower
|
|
// systems with a large set of manuals, but it should be ok since apropos is only called once.
|
|
let mut list = vec![];
|
|
exec_subshell(
|
|
&lookup_cmd,
|
|
parser,
|
|
Some(&mut list),
|
|
false, /* don't apply exit status */
|
|
);
|
|
|
|
// Then discard anything that is not a possible completion and put the result into a
|
|
// hashtable with the completion as key and the description as value.
|
|
let mut lookup = HashMap::new();
|
|
// A typical entry is the command name, followed by a tab, followed by a description.
|
|
for elstr in &mut list {
|
|
// Skip keys that are too short.
|
|
if elstr.len() < cmd.len() {
|
|
continue;
|
|
}
|
|
|
|
// Skip cases without a tab, or without a description, or bizarre cases where the tab is
|
|
// part of the command.
|
|
let Some(tab_idx) = elstr.find_char('\t') else {
|
|
continue;
|
|
};
|
|
if tab_idx + 1 >= elstr.len() || tab_idx < cmd.len() {
|
|
continue;
|
|
}
|
|
|
|
// Make the key. This is the stuff after the command.
|
|
// For example:
|
|
// elstr = lsmod
|
|
// cmd = ls
|
|
// key = mod
|
|
// Note an empty key is common and natural, if 'cmd' were already valid.
|
|
let (key, val) = elstr.split_at_mut(cmd.len());
|
|
let val = &mut val[1..];
|
|
assert!(
|
|
!val.is_empty(),
|
|
"tab index should not have been at the end."
|
|
);
|
|
|
|
// And once again I make sure the first character is uppercased because I like it that
|
|
// way, and I get to decide these things.
|
|
let mut upper_chars = val.as_char_slice()[0].to_uppercase();
|
|
if let (Some(c), None) = (upper_chars.next(), upper_chars.next()) {
|
|
val.as_char_slice_mut()[0] = c;
|
|
}
|
|
lookup.insert(&*key, &*val);
|
|
}
|
|
|
|
// Then do a lookup on every completion and if a match is found, change to the new
|
|
// description.
|
|
for completion in self.completions.get_list_mut() {
|
|
let el = &completion.completion;
|
|
if let Some(&desc) = lookup.get(el.as_utfstr()) {
|
|
completion.description = desc.to_owned();
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Complete the specified command name. Search for executables in the path, executables defined
|
|
/// using an absolute path, functions, builtins and directories for implicit cd commands.
|
|
///
|
|
/// \param str_cmd the command string to find completions for
|
|
fn complete_cmd(&mut self, str_cmd: WString) {
|
|
// Append all possible executables
|
|
let result = {
|
|
let expand_flags = self.expand_flags()
|
|
| ExpandFlags::SPECIAL_FOR_COMMAND
|
|
| ExpandFlags::FOR_COMPLETIONS
|
|
| ExpandFlags::PRESERVE_HOME_TILDES
|
|
| ExpandFlags::EXECUTABLES_ONLY;
|
|
expand_to_receiver(
|
|
str_cmd.clone(),
|
|
&mut self.completions,
|
|
expand_flags,
|
|
self.ctx,
|
|
None,
|
|
)
|
|
.result
|
|
};
|
|
if result == ExpandResultCode::cancel {
|
|
return;
|
|
}
|
|
if result == ExpandResultCode::ok && self.flags.descriptions {
|
|
self.complete_cmd_desc(&str_cmd);
|
|
}
|
|
|
|
// We don't really care if this succeeds or fails. If it succeeds this->completions will be
|
|
// updated with choices for the user.
|
|
let _ = {
|
|
// Append all matching directories
|
|
let expand_flags = self.expand_flags()
|
|
| ExpandFlags::FOR_COMPLETIONS
|
|
| ExpandFlags::PRESERVE_HOME_TILDES
|
|
| ExpandFlags::DIRECTORIES_ONLY;
|
|
expand_to_receiver(
|
|
str_cmd.clone(),
|
|
&mut self.completions,
|
|
expand_flags,
|
|
self.ctx,
|
|
None,
|
|
)
|
|
};
|
|
|
|
if str_cmd.is_empty() || (!str_cmd.contains('/') && str_cmd.as_char_slice()[0] != '~') {
|
|
let include_hidden = str_cmd.as_char_slice().first() == Some(&'_');
|
|
// Append all known matching functions
|
|
let possible_comp: Vec<_> = function::get_names(include_hidden)
|
|
.into_iter()
|
|
.map(Completion::from_completion)
|
|
.collect();
|
|
|
|
self.complete_strings(
|
|
&str_cmd,
|
|
&{ Box::new(complete_function_desc) as DescriptionFunc },
|
|
&possible_comp,
|
|
CompleteFlags::empty(),
|
|
ExpandFlags::empty(),
|
|
);
|
|
|
|
// Append all matching builtins
|
|
let possible_comp: Vec<_> = builtin_get_names()
|
|
.map(wstr::to_owned)
|
|
.map(Completion::from_completion)
|
|
.collect();
|
|
|
|
self.complete_strings(
|
|
&str_cmd,
|
|
&{ Box::new(|name| builtin_get_desc(name).unwrap_or(L!("")).to_owned()) },
|
|
&possible_comp,
|
|
CompleteFlags::empty(),
|
|
ExpandFlags::empty(),
|
|
);
|
|
}
|
|
}
|
|
|
|
/// Attempt to complete an abbreviation for the given string.
|
|
fn complete_abbr(&mut self, cmd: WString) {
|
|
// Copy the list of names and descriptions so as not to hold the lock across the call to
|
|
// complete_strings.
|
|
let mut possible_comp = Vec::new();
|
|
let mut descs = HashMap::new();
|
|
with_abbrs(|set| {
|
|
for abbr in set.list() {
|
|
if !abbr.is_regex() {
|
|
possible_comp.push(Completion::from_completion(abbr.key.clone()));
|
|
descs.insert(abbr.key.clone(), abbr.replacement.clone());
|
|
}
|
|
}
|
|
});
|
|
|
|
let desc_func = move |key: &wstr| {
|
|
let replacement = descs.get(key).expect("Abbreviation not found");
|
|
sprintf!(*ABBR_DESC, replacement)
|
|
};
|
|
self.complete_strings(
|
|
&cmd,
|
|
&{ Box::new(desc_func) as _ },
|
|
&possible_comp,
|
|
CompleteFlags::NO_SPACE,
|
|
ExpandFlags::empty(),
|
|
);
|
|
}
|
|
|
|
/// Evaluate the argument list (as supplied by `complete -a`) and insert any
|
|
/// return matching completions. Matching is done using `copy_strings_with_prefix`,
|
|
/// meaning the completion may contain wildcards.
|
|
/// Logically, this is not always the right thing to do, but I have yet to come
|
|
/// up with a case where this matters.
|
|
///
|
|
/// - `str`: The string to complete.
|
|
/// - `args`: The list of option arguments to be evaluated.
|
|
/// - `desc`: Description of the completion
|
|
/// - `flags`: The flags
|
|
fn complete_from_args(&mut self, s: &wstr, args: &wstr, desc: &wstr, flags: CompleteFlags) {
|
|
let is_autosuggest = self.flags.autosuggestion;
|
|
|
|
let saved_state = if let Some(parser) = self.ctx.maybe_parser() {
|
|
let saved_interactive = parser.libdata().pods.is_interactive;
|
|
parser.libdata_mut().pods.is_interactive = false;
|
|
|
|
Some((saved_interactive, parser.get_last_statuses()))
|
|
} else {
|
|
None
|
|
};
|
|
|
|
let eflags = if is_autosuggest {
|
|
ExpandFlags::SKIP_CMDSUBST
|
|
} else {
|
|
ExpandFlags::empty()
|
|
};
|
|
|
|
let possible_comp = Parser::expand_argument_list(args, eflags, self.ctx);
|
|
|
|
if let Some(parser) = self.ctx.maybe_parser() {
|
|
let (saved_interactive, status) = saved_state.unwrap();
|
|
parser.libdata_mut().pods.is_interactive = saved_interactive;
|
|
parser.set_last_statuses(status);
|
|
}
|
|
|
|
// Allow leading dots - see #3707.
|
|
self.complete_strings(
|
|
&escape(s),
|
|
&const_desc(desc),
|
|
&possible_comp,
|
|
flags,
|
|
ExpandFlags::ALLOW_NONLITERAL_LEADING_DOT,
|
|
);
|
|
}
|
|
|
|
/// complete_param: Given a command, find completions for the argument `s` of command `cmd_orig`
|
|
/// with previous option `popt`. If file completions should be disabled, then mark
|
|
/// `out_do_file` as `false`.
|
|
///
|
|
/// Returns `true` if successful, `false` if there's an error.
|
|
///
|
|
/// Examples in format (cmd, popt, str):
|
|
///
|
|
/// ```text
|
|
/// echo hello world <tab> -> ("echo", "world", "")
|
|
/// echo hello world<tab> -> ("echo", "hello", "world")
|
|
/// ```
|
|
fn complete_param_for_command(
|
|
&mut self,
|
|
cmd_orig: &wstr,
|
|
popt: &wstr,
|
|
s: &wstr,
|
|
use_switches: bool,
|
|
out_do_file: &mut bool,
|
|
) -> bool {
|
|
let mut use_files = true;
|
|
let mut has_force = false;
|
|
|
|
let CmdString { cmd, path } = parse_cmd_string(cmd_orig, self.ctx.vars());
|
|
|
|
// Don't use cmd_orig here for paths. It's potentially pathed,
|
|
// so that command might exist, but the completion script
|
|
// won't be using it.
|
|
let cmd_exists = builtin_exists(&cmd)
|
|
|| function::exists_no_autoload(&cmd)
|
|
|| path_get_path(&cmd, self.ctx.vars()).is_some();
|
|
if !cmd_exists {
|
|
// Do not load custom completions if the command does not exist
|
|
// This prevents errors caused during the execution of completion providers for
|
|
// tools that do not exist. Applies to both manual completions ("cm<TAB>", "cmd <TAB>")
|
|
// and automatic completions ("gi" autosuggestion provider -> git)
|
|
FLOG!(complete, "Skipping completions for non-existent command");
|
|
} else if let Some(parser) = self.ctx.maybe_parser() {
|
|
complete_load(&cmd, parser);
|
|
} else if !completion_autoloader
|
|
.lock()
|
|
.unwrap()
|
|
.has_attempted_autoload(&cmd)
|
|
{
|
|
self.needs_load.push(cmd.clone());
|
|
}
|
|
|
|
// Make a list of lists of all options that we care about.
|
|
let all_options: Vec<Vec<CompleteEntryOpt>> = COMPLETION_MAP
|
|
.lock()
|
|
.unwrap()
|
|
.iter()
|
|
.filter_map(|(idx, completion)| {
|
|
let r#match = if idx.is_path { &path } else { &cmd };
|
|
if wildcard_match(r#match, &idx.name, false) {
|
|
// Copy all of their options into our list. Oof, this is a lot of copying.
|
|
let mut options = completion.get_options().to_vec();
|
|
// We have to copy them in reverse order to preserve legacy behavior (#9221).
|
|
options.reverse();
|
|
Some(options)
|
|
} else {
|
|
None
|
|
}
|
|
})
|
|
.collect();
|
|
|
|
// Now release the lock and test each option that we captured above. We have to do this outside
|
|
// the lock because callouts (like the condition) may add or remove completions. See issue #2.
|
|
for options in all_options {
|
|
let short_opt_pos = short_option_pos(s, &options);
|
|
// We want last_option_requires_param to default to false but distinguish between when
|
|
// a previous completion has set it to false and when it has its default value.
|
|
let mut last_option_requires_param = None;
|
|
let mut use_common = true;
|
|
if use_switches {
|
|
if s.char_at(0) == '-' {
|
|
// Check if we are entering a combined option and argument (like --color=auto or
|
|
// -I/usr/include).
|
|
for o in &options {
|
|
let arg = if o.typ == CompleteOptionType::Short {
|
|
let Some(short_opt_pos) = short_opt_pos else {
|
|
continue;
|
|
};
|
|
if o.option.char_at(0) != s.char_at(short_opt_pos) {
|
|
continue;
|
|
}
|
|
Some(s.slice_from(short_opt_pos + 1))
|
|
} else {
|
|
param_match2(o, s)
|
|
};
|
|
|
|
if self.conditions_test(&o.conditions) {
|
|
if o.typ == CompleteOptionType::Short {
|
|
// Only override a true last_option_requires_param value with a false
|
|
// one
|
|
*last_option_requires_param
|
|
.get_or_insert(o.result_mode.requires_param) &=
|
|
o.result_mode.requires_param;
|
|
}
|
|
if let Some(arg) = arg {
|
|
if o.result_mode.requires_param {
|
|
use_common = false;
|
|
}
|
|
if o.result_mode.no_files {
|
|
use_files = false;
|
|
}
|
|
if o.result_mode.force_files {
|
|
has_force = true;
|
|
}
|
|
self.complete_from_args(arg, &o.comp, o.localized_desc(), o.flags);
|
|
}
|
|
}
|
|
}
|
|
} else if popt.char_at(0) == '-' {
|
|
// Set to true if we found a matching old-style switch.
|
|
// Here we are testing the previous argument,
|
|
// to see how we should complete the current argument
|
|
let mut old_style_match = false;
|
|
|
|
// If we are using old style long options, check for them first.
|
|
for o in &options {
|
|
if o.typ == CompleteOptionType::SingleLong
|
|
&& param_match(o, popt)
|
|
&& self.conditions_test(&o.conditions)
|
|
{
|
|
old_style_match = false;
|
|
if o.result_mode.requires_param {
|
|
use_common = false;
|
|
}
|
|
if o.result_mode.no_files {
|
|
use_files = false;
|
|
}
|
|
if o.result_mode.force_files {
|
|
has_force = true;
|
|
}
|
|
self.complete_from_args(s, &o.comp, o.localized_desc(), o.flags);
|
|
}
|
|
}
|
|
|
|
// No old style option matched, or we are not using old style options. We check if
|
|
// any short (or gnu style) options do.
|
|
if !old_style_match {
|
|
let prev_short_opt_pos = short_option_pos(popt, &options);
|
|
for o in &options {
|
|
// Gnu-style options with _optional_ arguments must be specified as a single
|
|
// token, so that it can be differed from a regular argument.
|
|
// Here we are testing the previous argument for a GNU-style match,
|
|
// to see how we should complete the current argument
|
|
if !o.result_mode.requires_param {
|
|
continue;
|
|
}
|
|
|
|
let mut r#match = false;
|
|
if o.typ == CompleteOptionType::Short {
|
|
if let Some(prev_short_opt_pos) = prev_short_opt_pos {
|
|
r#match = prev_short_opt_pos + 1 == popt.len()
|
|
&& o.option.char_at(0) == popt.char_at(prev_short_opt_pos);
|
|
}
|
|
} else if o.typ == CompleteOptionType::DoubleLong {
|
|
r#match = param_match(o, popt);
|
|
}
|
|
if r#match && self.conditions_test(&o.conditions) {
|
|
if o.result_mode.requires_param {
|
|
use_common = false;
|
|
}
|
|
if o.result_mode.no_files {
|
|
use_files = false;
|
|
}
|
|
if o.result_mode.force_files {
|
|
has_force = true;
|
|
}
|
|
self.complete_from_args(s, &o.comp, o.localized_desc(), o.flags);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if !use_common {
|
|
continue;
|
|
}
|
|
|
|
// Set a default value for last_option_requires_param only if one hasn't been set
|
|
let last_option_requires_param = last_option_requires_param.unwrap_or(false);
|
|
|
|
// Now we try to complete an option itself
|
|
for o in &options {
|
|
// If this entry is for the base command, check if any of the arguments match.
|
|
if !self.conditions_test(&o.conditions) {
|
|
continue;
|
|
}
|
|
if o.option.is_empty() {
|
|
use_files &= !o.result_mode.no_files;
|
|
has_force |= o.result_mode.force_files;
|
|
self.complete_from_args(s, &o.comp, o.localized_desc(), o.flags);
|
|
}
|
|
|
|
if !use_switches || s.is_empty() {
|
|
continue;
|
|
}
|
|
|
|
// Check if the short style option matches.
|
|
if o.typ == CompleteOptionType::Short {
|
|
let optchar = o.option.char_at(0);
|
|
if let Some(short_opt_pos) = short_opt_pos {
|
|
// Only complete when the last short option has no parameter yet..
|
|
if short_opt_pos + 1 != s.len() {
|
|
continue;
|
|
}
|
|
// .. and it does not require one ..
|
|
if last_option_requires_param {
|
|
continue;
|
|
}
|
|
// .. and the option is not already there.
|
|
if s.contains(optchar) {
|
|
continue;
|
|
}
|
|
} else {
|
|
// str has no short option at all (but perhaps it is the
|
|
// prefix of a single long option).
|
|
// Only complete short options if there is no character after the dash.
|
|
|
|
if s != L!("-") {
|
|
continue;
|
|
}
|
|
}
|
|
// It's a match.
|
|
let desc = o.localized_desc();
|
|
// Append a short-style option
|
|
if !self
|
|
.completions
|
|
.add(Completion::with_desc(o.option.clone(), desc.to_owned()))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Check if the long style option matches.
|
|
if o.typ != CompleteOptionType::SingleLong
|
|
&& o.typ != CompleteOptionType::DoubleLong
|
|
{
|
|
continue;
|
|
}
|
|
|
|
let whole_opt = L!("-").repeat(o.expected_dash_count()) + o.option.as_utfstr();
|
|
if whole_opt.len() < s.len() {
|
|
continue;
|
|
}
|
|
let r#match = string_prefixes_string(s, &whole_opt);
|
|
if !r#match {
|
|
let match_no_case = string_prefixes_string_case_insensitive(s, &whole_opt);
|
|
if !match_no_case {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
let mut offset = 0;
|
|
let mut flags = CompleteFlags::empty();
|
|
|
|
if r#match {
|
|
offset = s.len();
|
|
} else {
|
|
flags = CompleteFlags::REPLACES_TOKEN;
|
|
}
|
|
|
|
// does this switch have any known arguments
|
|
let has_arg = !o.comp.is_empty();
|
|
// does this switch _require_ an argument
|
|
let req_arg = o.result_mode.requires_param;
|
|
|
|
if o.typ == CompleteOptionType::DoubleLong && (has_arg && !req_arg) {
|
|
// Optional arguments to a switch can only be handled using the '=', so we add it as
|
|
// a completion. By default we avoid using '=' and instead rely on '--switch
|
|
// switch-arg', since it is more commonly supported by homebrew getopt-like
|
|
// functions.
|
|
let completion = sprintf!("%ls=", whole_opt.slice_from(offset));
|
|
|
|
// Append a long-style option with a mandatory trailing equal sign
|
|
if !self.completions.add(Completion::new(
|
|
completion,
|
|
o.localized_desc().to_owned(),
|
|
StringFuzzyMatch::exact_match(),
|
|
flags | CompleteFlags::NO_SPACE,
|
|
)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Append a long-style option
|
|
if !self.completions.add(Completion::new(
|
|
whole_opt.slice_from(offset).to_owned(),
|
|
o.localized_desc().to_owned(),
|
|
StringFuzzyMatch::exact_match(),
|
|
flags,
|
|
)) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if has_force {
|
|
*out_do_file = true;
|
|
} else if !use_files {
|
|
*out_do_file = false;
|
|
}
|
|
|
|
true
|
|
}
|
|
|
|
/// Perform generic (not command-specific) expansions on the specified string.
|
|
fn complete_param_expand(&mut self, s: &wstr, do_file: bool, handle_as_special_cd: bool) {
|
|
if self.ctx.check_cancel() {
|
|
return;
|
|
}
|
|
let mut flags = self.expand_flags()
|
|
| ExpandFlags::SKIP_CMDSUBST
|
|
| ExpandFlags::FOR_COMPLETIONS
|
|
| ExpandFlags::PRESERVE_HOME_TILDES;
|
|
if !do_file {
|
|
flags |= ExpandFlags::SKIP_WILDCARDS;
|
|
}
|
|
|
|
if handle_as_special_cd && do_file {
|
|
if self.flags.autosuggestion {
|
|
flags |= ExpandFlags::SPECIAL_FOR_CD_AUTOSUGGESTION;
|
|
}
|
|
flags |= ExpandFlags::DIRECTORIES_ONLY;
|
|
flags |= ExpandFlags::SPECIAL_FOR_CD;
|
|
}
|
|
|
|
// Squelch file descriptions per issue #254.
|
|
if self.flags.autosuggestion || do_file {
|
|
flags -= ExpandFlags::GEN_DESCRIPTIONS;
|
|
}
|
|
|
|
// We have the following cases:
|
|
//
|
|
// --foo=bar => expand just bar
|
|
// -foo=bar => expand just bar
|
|
// foo=bar => expand the whole thing, and also just bar
|
|
//
|
|
// We also support colon separator (#2178). If there's more than one, prefer the last one.
|
|
let sep_index = s.chars().rposition(|c| c == '=' || c == ':');
|
|
let complete_from_start = sep_index.is_none() || !string_prefixes_string(L!("-"), s);
|
|
|
|
if let Some(sep_index) = sep_index {
|
|
// FIXME: This just cuts the token,
|
|
// so any quoting or braces gets lost.
|
|
// See #4954.
|
|
let sep_string = s.slice_from(sep_index + 1);
|
|
let mut local_completions = Vec::new();
|
|
if expand_string(
|
|
sep_string.to_owned(),
|
|
&mut local_completions,
|
|
flags,
|
|
self.ctx,
|
|
None,
|
|
)
|
|
.result
|
|
== ExpandResultCode::error
|
|
{
|
|
FLOGF!(complete, "Error while expanding string '%ls'", sep_string);
|
|
}
|
|
|
|
// Any COMPLETE_REPLACES_TOKEN will also stomp the separator. We need to "repair" them by
|
|
// inserting our separator and prefix.
|
|
let prefix_with_sep = s.as_char_slice()[..sep_index + 1].into();
|
|
for comp in &mut local_completions {
|
|
comp.prepend_token_prefix(prefix_with_sep);
|
|
}
|
|
if !self.completions.extend(local_completions) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if complete_from_start {
|
|
// Don't do fuzzy matching for files if the string begins with a dash (issue #568). We could
|
|
// consider relaxing this if there was a preceding double-dash argument.
|
|
if string_prefixes_string(L!("-"), s) {
|
|
flags -= ExpandFlags::FUZZY_MATCH;
|
|
}
|
|
|
|
if expand_to_receiver(s.to_owned(), &mut self.completions, flags, self.ctx, None).result
|
|
== ExpandResultCode::error
|
|
{
|
|
FLOGF!(complete, "Error while expanding string '%ls'", s);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Complete the specified string as an environment variable.
|
|
/// Returns `true` if this was a variable, so we should stop completion.
|
|
fn complete_variable(&mut self, s: &wstr, start_offset: usize) -> bool {
|
|
let whole_var = s;
|
|
let var = whole_var.slice_from(start_offset);
|
|
let varlen = s.len() - start_offset;
|
|
let mut res = false;
|
|
|
|
for env_name in self.ctx.vars().get_names(EnvMode::empty()) {
|
|
let anchor_start = !self.flags.fuzzy_match;
|
|
let Some(r#match) = string_fuzzy_match_string(var, &env_name, anchor_start) else {
|
|
continue;
|
|
};
|
|
|
|
let (comp, flags) = if !r#match.requires_full_replacement() {
|
|
// Take only the suffix.
|
|
(
|
|
env_name.slice_from(varlen).to_owned(),
|
|
CompleteFlags::empty(),
|
|
)
|
|
} else {
|
|
let comp = whole_var.slice_to(start_offset).to_owned() + env_name.as_utfstr();
|
|
let flags = CompleteFlags::REPLACES_TOKEN | CompleteFlags::DONT_ESCAPE;
|
|
(comp, flags)
|
|
};
|
|
|
|
let mut desc = WString::new();
|
|
if self.flags.descriptions && self.flags.autosuggestion {
|
|
// $history can be huge, don't put all of it in the completion description; see
|
|
// #6288.
|
|
if env_name == L!("history") {
|
|
let history = History::with_name(&history_session_id(self.ctx.vars()));
|
|
for i in 1..std::cmp::min(history.size(), 64) {
|
|
if i > 1 {
|
|
desc.push(' ');
|
|
}
|
|
desc.push_utfstr(&expand_escape_string(
|
|
history.item_at_index(i).unwrap().str(),
|
|
));
|
|
}
|
|
} else {
|
|
// Can't use ctx.vars() here, it could be any variable.
|
|
let Some(var) = self.ctx.vars().get(&env_name) else {
|
|
continue;
|
|
};
|
|
|
|
let value = expand_escape_variable(&var);
|
|
desc = sprintf!(*COMPLETE_VAR_DESC_VAL, value);
|
|
}
|
|
}
|
|
|
|
// Append matching environment variables
|
|
// TODO: need to propagate overflow here.
|
|
let _ = self
|
|
.completions
|
|
.add(Completion::new(comp, desc, r#match, flags));
|
|
|
|
res = true;
|
|
}
|
|
|
|
res
|
|
}
|
|
|
|
fn try_complete_variable(&mut self, s: &wstr) -> bool {
|
|
#[derive(PartialEq, Eq)]
|
|
enum Mode {
|
|
Unquoted,
|
|
SingleQuoted,
|
|
DoubleQuoted,
|
|
}
|
|
use Mode::*;
|
|
|
|
let mut mode = Unquoted;
|
|
|
|
// Get the position of the dollar heading a (possibly empty) run of valid variable characters.
|
|
let mut variable_start = None;
|
|
|
|
let mut skip_next = false;
|
|
for (in_pos, c) in s.chars().enumerate() {
|
|
if skip_next {
|
|
skip_next = false;
|
|
continue;
|
|
}
|
|
|
|
if !valid_var_name_char(c) {
|
|
// This character cannot be in a variable, reset the dollar.
|
|
variable_start = None;
|
|
}
|
|
|
|
match c {
|
|
'\\' => skip_next = true,
|
|
'$' => {
|
|
if mode == Unquoted || mode == DoubleQuoted {
|
|
variable_start = Some(in_pos);
|
|
}
|
|
}
|
|
'\'' => {
|
|
if mode == SingleQuoted {
|
|
mode = Unquoted;
|
|
} else if mode == Unquoted {
|
|
mode = SingleQuoted;
|
|
}
|
|
}
|
|
'"' => {
|
|
if mode == DoubleQuoted {
|
|
mode = Unquoted;
|
|
} else if mode == Unquoted {
|
|
mode = DoubleQuoted;
|
|
}
|
|
}
|
|
_ => {
|
|
// all other chars ignored here
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now complete if we have a variable start. Note the variable text may be empty; in that case
|
|
// don't generate an autosuggestion, but do allow tab completion.
|
|
let allow_empty = !self.flags.autosuggestion;
|
|
let text_is_empty = variable_start == Some(s.len() - 1);
|
|
if let Some(variable_start) = variable_start {
|
|
if allow_empty || !text_is_empty {
|
|
return self.complete_variable(s, variable_start + 1);
|
|
}
|
|
}
|
|
|
|
false
|
|
}
|
|
|
|
/// Try to complete the specified string as a username. This is used by `~USER` type expansion.
|
|
///
|
|
/// Returns `false` if unable to complete, `true` otherwise
|
|
fn try_complete_user(&mut self, s: &wstr) -> bool {
|
|
#[cfg(target_os = "android")]
|
|
{
|
|
// The getpwent() function does not exist on Android. A Linux user on Android isn't
|
|
// really a user - each installed app gets an UID assigned. Listing all UID:s is not
|
|
// possible without root access, and doing a ~USER type expansion does not make sense
|
|
// since every app is sandboxed and can't access eachother.
|
|
return false;
|
|
}
|
|
#[cfg(not(target_os = "android"))]
|
|
{
|
|
static s_setpwent_lock: Mutex<()> = Mutex::new(());
|
|
|
|
if s.char_at(0) != '~' || s.contains('/') {
|
|
return false;
|
|
}
|
|
|
|
let user_name = s.slice_from(1);
|
|
if user_name.contains('~') {
|
|
return false;
|
|
}
|
|
|
|
let start_time = Instant::now();
|
|
let mut result = false;
|
|
let name_len = s.len() - 1;
|
|
|
|
fn getpwent_name() -> Option<WString> {
|
|
let ptr = unsafe { libc::getpwent() };
|
|
if ptr.is_null() {
|
|
return None;
|
|
}
|
|
let pw = unsafe { ptr.read() };
|
|
Some(charptr2wcstring(pw.pw_name))
|
|
}
|
|
|
|
let _guard = s_setpwent_lock.lock().unwrap();
|
|
|
|
unsafe { libc::setpwent() };
|
|
while let Some(pw_name) = getpwent_name() {
|
|
if self.ctx.check_cancel() {
|
|
break;
|
|
}
|
|
|
|
if string_prefixes_string(user_name, &pw_name) {
|
|
let desc = sprintf!(*COMPLETE_USER_DESC, &pw_name);
|
|
// Append a user name.
|
|
// TODO: propagate overflow?
|
|
let _ = self.completions.add(Completion::new(
|
|
pw_name.slice_from(name_len).to_owned(),
|
|
desc,
|
|
StringFuzzyMatch::exact_match(),
|
|
CompleteFlags::NO_SPACE,
|
|
));
|
|
result = true;
|
|
} else if string_prefixes_string_case_insensitive(user_name, &pw_name) {
|
|
let name = sprintf!("~%ls", &pw_name);
|
|
let desc = sprintf!(*COMPLETE_USER_DESC, &pw_name);
|
|
|
|
// Append a user name
|
|
// TODO: propagate overflow?
|
|
let _ = self.completions.add(Completion::new(
|
|
name,
|
|
desc,
|
|
StringFuzzyMatch::exact_match(),
|
|
CompleteFlags::REPLACES_TOKEN
|
|
| CompleteFlags::DONT_ESCAPE
|
|
| CompleteFlags::NO_SPACE,
|
|
));
|
|
result = true;
|
|
}
|
|
|
|
// If we've spent too much time (more than 200 ms) doing this give up.
|
|
if start_time.elapsed() > Duration::from_millis(200) {
|
|
break;
|
|
}
|
|
}
|
|
unsafe { libc::endpwent() };
|
|
|
|
result
|
|
}
|
|
}
|
|
|
|
/// If we have variable assignments, attempt to apply them in our parser. As soon as the return
|
|
/// value goes out of scope, the variables will be removed from the parser.
|
|
fn apply_var_assignments<'a>(
|
|
&mut self,
|
|
var_assignments: impl IntoIterator<Item = &'a wstr>,
|
|
) -> Option<ScopeGuard<(), impl FnOnce(&mut ())>> {
|
|
if !self.ctx.has_parser() {
|
|
return None;
|
|
}
|
|
let parser = self.ctx.parser();
|
|
|
|
let mut var_assignments = var_assignments.into_iter().peekable();
|
|
var_assignments.peek()?;
|
|
|
|
let vars = parser.vars();
|
|
assert_eq!(
|
|
self.ctx.vars() as *const _ as *const (),
|
|
vars as *const _ as *const (),
|
|
"Don't know how to tab complete with a parser but a different variable set"
|
|
);
|
|
|
|
// clone of parse_execution_context_t::apply_variable_assignments.
|
|
// Crucially do NOT expand subcommands:
|
|
// VAR=(launch_missiles) cmd<tab>
|
|
// should not launch missiles.
|
|
// Note we also do NOT send --on-variable events.
|
|
let expand_flags = ExpandFlags::SKIP_CMDSUBST;
|
|
let block = parser.push_block(Block::variable_assignment_block());
|
|
for var_assign in var_assignments {
|
|
let equals_pos = variable_assignment_equals_pos(var_assign)
|
|
.expect("All variable assignments should have equals position");
|
|
let variable_name = var_assign.as_char_slice()[..equals_pos].into();
|
|
let expression = var_assign.slice_from(equals_pos + 1);
|
|
|
|
let mut expression_expanded = Vec::new();
|
|
let expand_ret = expand_string(
|
|
expression.to_owned(),
|
|
&mut expression_expanded,
|
|
expand_flags,
|
|
self.ctx,
|
|
None,
|
|
);
|
|
// If expansion succeeds, set the value; if it fails (e.g. it has a cmdsub) set an empty
|
|
// value anyways.
|
|
let vals = if expand_ret.result == ExpandResultCode::ok {
|
|
expression_expanded
|
|
.into_iter()
|
|
.map(|c| c.completion)
|
|
.collect()
|
|
} else {
|
|
Vec::new()
|
|
};
|
|
parser
|
|
.vars()
|
|
.set(variable_name, EnvMode::LOCAL | EnvMode::EXPORT, vals);
|
|
if self.ctx.check_cancel() {
|
|
break;
|
|
}
|
|
}
|
|
|
|
let parser_ref = self.ctx.parser().shared();
|
|
Some(ScopeGuard::new((), move |_| parser_ref.pop_block(block)))
|
|
}
|
|
|
|
/// Complete a command by invoking user-specified completions.
|
|
fn complete_custom(&mut self, cmd: &wstr, cmdline: &wstr, ad: &mut CustomArgData) {
|
|
if self.ctx.check_cancel() {
|
|
return;
|
|
}
|
|
|
|
let is_autosuggest = self.flags.autosuggestion;
|
|
// Perhaps set a transient commandline so that custom completions
|
|
// builtin_commandline will refer to the wrapped command. But not if
|
|
// we're doing autosuggestions.
|
|
let mut _remove_transient = None;
|
|
let wants_transient =
|
|
(ad.wrap_depth > 0 || !ad.var_assignments.is_empty()) && !is_autosuggest;
|
|
if wants_transient {
|
|
let parser_ref = self.ctx.parser().shared();
|
|
parser_ref
|
|
.libdata_mut()
|
|
.transient_commandlines
|
|
.push(cmdline.to_owned());
|
|
_remove_transient = Some(ScopeGuard::new((), move |_| {
|
|
parser_ref.libdata_mut().transient_commandlines.pop();
|
|
}));
|
|
}
|
|
|
|
// Maybe apply variable assignments.
|
|
let _restore_vars =
|
|
self.apply_var_assignments(ad.var_assignments.iter().map(WString::as_utfstr));
|
|
if self.ctx.check_cancel() {
|
|
return;
|
|
}
|
|
|
|
// Invoke any custom completions for this command.
|
|
self.complete_param_for_command(
|
|
cmd,
|
|
&ad.previous_argument,
|
|
&ad.current_argument,
|
|
!ad.had_ddash,
|
|
&mut ad.do_file,
|
|
);
|
|
}
|
|
|
|
// Invoke command-specific completions given by \p arg_data.
|
|
// Then, for each target wrapped by the given command, update the command
|
|
// line with that target and invoke this recursively.
|
|
// The command whose completions to use is given by \p cmd. The full command line is given by \p
|
|
// cmdline and the command's range in it is given by \p cmdrange. Note: the command range
|
|
// may have a different length than the command itself, because the command is unescaped (i.e.
|
|
// quotes removed).
|
|
fn walk_wrap_chain(
|
|
&mut self,
|
|
cmd: &wstr,
|
|
cmdline: &wstr,
|
|
cmdrange: SourceRange,
|
|
ad: &mut CustomArgData,
|
|
) {
|
|
// Limit our recursion depth. This prevents cycles in the wrap chain graph from overflowing.
|
|
if ad.wrap_depth > 24 {
|
|
return;
|
|
}
|
|
if self.ctx.check_cancel() {
|
|
return;
|
|
}
|
|
|
|
// Extract command from the command line and invoke the receiver with it.
|
|
self.complete_custom(cmd, cmdline, ad);
|
|
|
|
let targets = complete_get_wrap_targets(cmd);
|
|
let wrap_depth = ad.wrap_depth;
|
|
let mut ad = ScopeGuard::new(ad, |ad| ad.wrap_depth = wrap_depth);
|
|
ad.wrap_depth += 1;
|
|
|
|
for wt in targets {
|
|
// We may append to the variable assignment list; ensure we restore it.
|
|
let saved_var_count = ad.var_assignments.len();
|
|
let mut ad = ScopeGuard::new(&mut ad, |ad| {
|
|
assert!(
|
|
ad.var_assignments.len() >= saved_var_count,
|
|
"Should not delete var assignments"
|
|
);
|
|
ad.var_assignments.truncate(saved_var_count);
|
|
});
|
|
|
|
// Separate the wrap target into any variable assignments VAR=... and the command itself.
|
|
let mut wrapped_command = None;
|
|
let mut wrapped_command_offset_in_wt = None;
|
|
let tokenizer = Tokenizer::new(&wt, TokFlags(0));
|
|
for tok in tokenizer {
|
|
let mut tok_src = tok.get_source(&wt).to_owned();
|
|
if variable_assignment_equals_pos(&tok_src).is_some() {
|
|
ad.var_assignments.push(tok_src);
|
|
} else {
|
|
expand_command_token(self.ctx, &mut tok_src);
|
|
|
|
wrapped_command_offset_in_wt = Some(tok.offset());
|
|
wrapped_command = Some(tok_src);
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Skip this wrapped command if empty, or if we've seen it before.
|
|
let Some((wrapped_command, wrapped_command_offset_in_wt)) =
|
|
Option::zip(wrapped_command, wrapped_command_offset_in_wt)
|
|
else {
|
|
continue;
|
|
};
|
|
|
|
if !ad.visited_wrapped_commands.insert(wrapped_command.clone()) {
|
|
continue;
|
|
}
|
|
|
|
// Construct a fake command line containing the wrap target.
|
|
// https://github.com/starkat99/widestring-rs/issues/37
|
|
let mut faux_commandline = cmdline.as_char_slice().to_vec();
|
|
faux_commandline.splice(std::ops::Range::from(cmdrange), wt.chars());
|
|
let faux_commandline = WString::from(faux_commandline);
|
|
|
|
// Recurse with our new command and command line.
|
|
let faux_source_range = SourceRange::new(
|
|
cmdrange.start() + wrapped_command_offset_in_wt,
|
|
wrapped_command.len(),
|
|
);
|
|
self.walk_wrap_chain(
|
|
&wrapped_command,
|
|
&faux_commandline,
|
|
faux_source_range,
|
|
***ad,
|
|
);
|
|
}
|
|
}
|
|
|
|
/// If the argument contains a '[' typed by the user, completion by appending to the argument might
|
|
/// produce an invalid token (#5831).
|
|
///
|
|
/// Check if there is any unescaped, unquoted '['; if yes, make the completions replace the entire
|
|
/// argument instead of appending, so '[' will be escaped.
|
|
fn escape_opening_brackets(&mut self, argument: &wstr) {
|
|
let mut have_unquoted_unescaped_bracket = false;
|
|
let mut quote = None;
|
|
let mut escaped = false;
|
|
for c in argument.chars() {
|
|
have_unquoted_unescaped_bracket |= c == '[' && quote.is_none() && !escaped;
|
|
if escaped {
|
|
escaped = false;
|
|
} else if c == '\\' {
|
|
escaped = true;
|
|
} else if c == '\'' || c == '"' {
|
|
if quote == Some(c) {
|
|
// Closing a quote.
|
|
quote = None;
|
|
} else if quote.is_none() {
|
|
// Opening a quote.
|
|
quote = Some(c);
|
|
}
|
|
}
|
|
}
|
|
if !have_unquoted_unescaped_bracket {
|
|
return;
|
|
}
|
|
|
|
// Since completion_apply_to_command_line will escape the completion, we need to provide an
|
|
// unescaped version.
|
|
let Some(unescaped_argument) = unescape_string(
|
|
argument,
|
|
UnescapeStringStyle::Script(UnescapeFlags::INCOMPLETE),
|
|
) else {
|
|
return;
|
|
};
|
|
for comp in self.completions.get_list_mut() {
|
|
if comp.flags.contains(CompleteFlags::REPLACES_TOKEN) {
|
|
continue;
|
|
}
|
|
comp.flags |= CompleteFlags::REPLACES_TOKEN;
|
|
comp.flags |= CompleteFlags::DONT_ESCAPE_TILDES; // See #9073.
|
|
|
|
// We are grafting a completion that is expected to be escaped later. This will break
|
|
// if the original completion doesn't want escaping. Happily, this is only the case
|
|
// for username completion and variable name completion. They shouldn't end up here
|
|
// anyway because they won't contain '['.
|
|
if comp.flags.contains(CompleteFlags::DONT_ESCAPE) {
|
|
FLOG!(warning, "unexpected completion flag");
|
|
}
|
|
comp.completion.insert_utfstr(0, &unescaped_argument);
|
|
}
|
|
}
|
|
|
|
/// Set the `DUPLICATES_ARG` flag in any completion that duplicates an argument.
|
|
fn mark_completions_duplicating_arguments(
|
|
&mut self,
|
|
cmd: &wstr,
|
|
prefix: &wstr,
|
|
args: impl IntoIterator<Item = Tok>,
|
|
) {
|
|
// Get all the arguments, unescaped, into an array that we're going to bsearch.
|
|
let mut arg_strs: Vec<_> = args
|
|
.into_iter()
|
|
.map(|arg| arg.get_source(cmd))
|
|
.filter_map(|argstr| unescape_string(argstr, UnescapeStringStyle::default()))
|
|
.collect();
|
|
arg_strs.sort();
|
|
|
|
let mut comp_str;
|
|
for comp in self.completions.get_list_mut() {
|
|
comp_str = comp.completion.clone();
|
|
if !comp.flags.contains(CompleteFlags::REPLACES_TOKEN) {
|
|
comp_str.insert_utfstr(0, prefix);
|
|
}
|
|
if arg_strs.binary_search(&comp_str).is_ok() {
|
|
comp.flags |= CompleteFlags::DUPLICATES_ARGUMENT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
struct CmdString {
|
|
cmd: WString,
|
|
path: WString,
|
|
}
|
|
|
|
/// Find the full path and commandname from a command string `s`.
|
|
fn parse_cmd_string(s: &wstr, vars: &dyn Environment) -> CmdString {
|
|
let path_result = path_try_get_path(s, vars);
|
|
let found = path_result.err.is_none();
|
|
let mut path = path_result.path;
|
|
// Resolve commands that use relative paths because we compare full paths with "complete -p".
|
|
if found && !path.is_empty() && path.as_char_slice().first() != Some(&'/') {
|
|
if let Some(full_path) = wrealpath(&path) {
|
|
path = full_path;
|
|
}
|
|
}
|
|
|
|
// Make sure the path is not included in the command.
|
|
let cmd = if let Some(last_slash) = s.chars().rposition(|c| c == '/') {
|
|
&s[last_slash + 1..]
|
|
} else {
|
|
s
|
|
}
|
|
.to_owned();
|
|
|
|
CmdString { cmd, path }
|
|
}
|
|
|
|
/// Returns a description for the specified function, or an empty string if none.
|
|
fn complete_function_desc(f: &wstr) -> WString {
|
|
if let Some(props) = function::get_props(f) {
|
|
props.description.clone()
|
|
} else {
|
|
WString::new()
|
|
}
|
|
}
|
|
|
|
fn leading_dash_count(s: &wstr) -> usize {
|
|
s.chars().take_while(|&c| c == '-').count()
|
|
}
|
|
|
|
/// Match a parameter.
|
|
fn param_match(e: &CompleteEntryOpt, optstr: &wstr) -> bool {
|
|
if e.typ == CompleteOptionType::ArgsOnly {
|
|
false
|
|
} else {
|
|
let dashes = leading_dash_count(optstr);
|
|
dashes == e.expected_dash_count() && e.option == optstr[dashes..]
|
|
}
|
|
}
|
|
|
|
/// Test if a string is an option with an argument, like --color=auto or -I/usr/include.
|
|
fn param_match2<'s>(e: &CompleteEntryOpt, optstr: &'s wstr) -> Option<&'s wstr> {
|
|
// We may get a complete_entry_opt_t with no options if it's just arguments.
|
|
if e.option.is_empty() {
|
|
return None;
|
|
}
|
|
|
|
// Verify leading dashes.
|
|
let mut cursor = leading_dash_count(optstr);
|
|
if cursor != e.expected_dash_count() {
|
|
return None;
|
|
}
|
|
|
|
// Verify options match.
|
|
if !optstr.slice_from(cursor).starts_with(&e.option) {
|
|
return None;
|
|
}
|
|
cursor += e.option.len();
|
|
|
|
// Short options are like -DNDEBUG. Long options are like --color=auto. So check for an equal
|
|
// sign for long options.
|
|
assert!(e.typ != CompleteOptionType::Short);
|
|
if optstr.char_at(cursor) != '=' {
|
|
return None;
|
|
}
|
|
cursor += 1;
|
|
Some(optstr.slice_from(cursor))
|
|
}
|
|
|
|
/// Parses a token of short options plus one optional parameter like
|
|
/// '-xzPARAM', where x and z are short options.
|
|
///
|
|
/// Returns the position of the last option character (e.g. the position of z which is 2).
|
|
/// Everything after that is assumed to be part of the parameter.
|
|
/// Returns wcstring::npos if there is no valid short option.
|
|
fn short_option_pos(arg: &wstr, options: &[CompleteEntryOpt]) -> Option<usize> {
|
|
if arg.len() <= 1 || leading_dash_count(arg) != 1 {
|
|
return None;
|
|
}
|
|
for (pos, arg_char) in arg.chars().enumerate().skip(1) {
|
|
let r#match = options
|
|
.iter()
|
|
.find(|o| o.typ == CompleteOptionType::Short && o.option.char_at(0) == arg_char);
|
|
if let Some(r#match) = r#match {
|
|
if r#match.result_mode.requires_param {
|
|
return Some(pos);
|
|
}
|
|
} else {
|
|
// The first character after the dash is not a valid option.
|
|
if pos == 1 {
|
|
return None;
|
|
}
|
|
return Some(pos - 1);
|
|
}
|
|
}
|
|
|
|
Some(arg.len() - 1)
|
|
}
|
|
|
|
fn expand_command_token(ctx: &OperationContext<'_>, cmd_tok: &mut WString) -> bool {
|
|
// TODO: we give up if the first token expands to more than one argument. We could handle
|
|
// that case by propagating arguments.
|
|
// Also we could expand wildcards.
|
|
expand_one(
|
|
cmd_tok,
|
|
ExpandFlags::SKIP_CMDSUBST | ExpandFlags::SKIP_WILDCARDS,
|
|
ctx,
|
|
None,
|
|
)
|
|
}
|
|
|
|
/// Create a new completion entry.
|
|
///
|
|
/// \param completions The array of completions to append to
|
|
/// \param comp The completion string
|
|
/// \param desc The description of the completion
|
|
/// \param flags completion flags
|
|
#[deprecated = "Use Vec::push()"]
|
|
pub fn append_completion(
|
|
completions: &mut Vec<Completion>,
|
|
comp: WString,
|
|
desc: WString,
|
|
flags: CompleteFlags,
|
|
r#match: StringFuzzyMatch,
|
|
) {
|
|
completions.push(Completion::new(comp, desc, r#match, flags))
|
|
}
|
|
|
|
/// Add an unexpanded completion "rule" to generate completions from for a command.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// The command 'gcc -o' requires that a file follows it, so the `requires_param` mode is suitable.
|
|
/// This can be done using the following line:
|
|
///
|
|
/// ```
|
|
/// complete -c gcc -s o -r
|
|
/// ```
|
|
///
|
|
/// The command 'grep -d' required that one of the strings 'read', 'skip' or 'recurse' is used. As
|
|
/// such, it is suitable to specify that a completion requires one of them. This can be done using
|
|
/// the following line:
|
|
///
|
|
/// ```
|
|
/// complete -c grep -s d -x -a "read skip recurse"
|
|
/// ```
|
|
///
|
|
/// - `cmd`: Command to complete.
|
|
/// - `cmd_is_path`: If `true`, cmd will be interpreted as the absolute
|
|
/// path of the program (optionally containing wildcards), otherwise it
|
|
/// will be interpreted as the command name.
|
|
/// - `option`: The name of an option.
|
|
/// - `option_type`: The type of option: can be option_type_short (-x),
|
|
/// option_type_single_long (-foo), option_type_double_long (--bar).
|
|
/// - `result_mode`: Controls how to search further completions when this completion has been
|
|
/// successfully matched.
|
|
/// - `comp`: A space separated list of completions which may contain subshells.
|
|
/// - `desc`: A description of the completion.
|
|
/// - `condition`: a command to be run to check it this completion should be used. If `condition`
|
|
/// is empty, the completion is always used.
|
|
/// - `flags`: A set of completion flags
|
|
#[allow(clippy::too_many_arguments)]
|
|
pub fn complete_add(
|
|
cmd: WString,
|
|
cmd_is_path: bool,
|
|
option: WString,
|
|
option_type: CompleteOptionType,
|
|
result_mode: CompletionMode,
|
|
condition: Vec<WString>,
|
|
comp: WString,
|
|
desc: WString,
|
|
flags: CompleteFlags,
|
|
) {
|
|
// option should be empty iff the option type is arguments only.
|
|
assert!(option.is_empty() == (option_type == CompleteOptionType::ArgsOnly));
|
|
|
|
// Lock the lock that allows us to edit the completion entry list.
|
|
let mut completion_map = COMPLETION_MAP.lock().expect("mutex poisoned");
|
|
let c = &mut completion_map
|
|
.entry(CompletionEntryIndex {
|
|
name: cmd,
|
|
is_path: cmd_is_path,
|
|
})
|
|
.or_insert_with(CompletionEntry::new);
|
|
|
|
// Create our new option.
|
|
let opt = CompleteEntryOpt {
|
|
option,
|
|
typ: option_type,
|
|
result_mode,
|
|
comp,
|
|
desc,
|
|
conditions: condition,
|
|
flags,
|
|
};
|
|
c.add_option(opt);
|
|
}
|
|
|
|
/// Remove a previously defined completion.
|
|
pub fn complete_remove(cmd: WString, cmd_is_path: bool, option: &wstr, typ: CompleteOptionType) {
|
|
let mut completion_map = COMPLETION_MAP.lock().expect("mutex poisoned");
|
|
let idx = CompletionEntryIndex {
|
|
name: cmd,
|
|
is_path: cmd_is_path,
|
|
};
|
|
if let Some(c) = completion_map.get_mut(&idx) {
|
|
let delete_it = c.remove_option(option, typ);
|
|
if delete_it {
|
|
completion_map.remove(&idx);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Removes all completions for a given command.
|
|
pub fn complete_remove_all(cmd: WString, cmd_is_path: bool) {
|
|
let mut completion_map = COMPLETION_MAP.lock().expect("mutex poisoned");
|
|
completion_map.remove(&CompletionEntryIndex {
|
|
name: cmd,
|
|
is_path: cmd_is_path,
|
|
});
|
|
}
|
|
|
|
/// Returns all completions of the command cmd.
|
|
/// If `ctx` contains a parser, this will autoload functions and completions as needed.
|
|
/// If it does not contain a parser, then any completions which need autoloading will be returned.
|
|
pub fn complete(
|
|
cmd_with_subcmds: &wstr,
|
|
flags: CompletionRequestOptions,
|
|
ctx: &OperationContext,
|
|
) -> (Vec<Completion>, Vec<WString>) {
|
|
// Determine the innermost subcommand.
|
|
let cmdsubst = parse_util_cmdsubst_extent(cmd_with_subcmds, cmd_with_subcmds.len());
|
|
let cmd = cmd_with_subcmds[cmdsubst].to_owned();
|
|
let mut completer = Completer::new(ctx, flags);
|
|
completer.perform_for_commandline(cmd);
|
|
|
|
(
|
|
completer.acquire_completions(),
|
|
completer.acquire_needs_load(),
|
|
)
|
|
}
|
|
|
|
/// Print the short switch `opt`, and the argument `arg` to the specified
|
|
/// [`WString`], but only if `argument` isn't an empty string.
|
|
fn append_switch_short_arg(out: &mut WString, opt: char, arg: &wstr) {
|
|
if arg.is_empty() {
|
|
return;
|
|
}
|
|
|
|
sprintf!(=> out, " -%lc %ls", opt, escape(arg));
|
|
}
|
|
fn append_switch_long_arg(out: &mut WString, opt: &wstr, arg: &wstr) {
|
|
if arg.is_empty() {
|
|
return;
|
|
}
|
|
|
|
sprintf!(=> out, " --%ls %ls", opt, escape(arg));
|
|
}
|
|
fn append_switch_short(out: &mut WString, opt: char) {
|
|
sprintf!(=> out, " -%lc", opt);
|
|
}
|
|
fn append_switch_long(out: &mut WString, opt: &wstr) {
|
|
sprintf!(=> out, " --%ls", opt);
|
|
}
|
|
|
|
fn completion2string(index: &CompletionEntryIndex, o: &CompleteEntryOpt) -> WString {
|
|
let mut out = WString::from(L!("complete"));
|
|
|
|
if o.flags.contains(CompleteFlags::DONT_SORT) {
|
|
append_switch_short(&mut out, 'k');
|
|
}
|
|
|
|
if o.result_mode.no_files && o.result_mode.requires_param {
|
|
append_switch_long(&mut out, L!("exclusive"));
|
|
} else if o.result_mode.no_files {
|
|
append_switch_long(&mut out, L!("no-files"));
|
|
} else if o.result_mode.force_files {
|
|
append_switch_long(&mut out, L!("force-files"));
|
|
} else if o.result_mode.requires_param {
|
|
append_switch_long(&mut out, L!("require-parameter"));
|
|
}
|
|
|
|
if index.is_path {
|
|
append_switch_short_arg(&mut out, 'p', &index.name);
|
|
} else {
|
|
out.push(' ');
|
|
out.push_utfstr(&escape(&index.name));
|
|
}
|
|
|
|
match o.typ {
|
|
CompleteOptionType::ArgsOnly => {}
|
|
CompleteOptionType::Short => append_switch_short_arg(&mut out, 's', &o.option[..1]),
|
|
CompleteOptionType::SingleLong => append_switch_short_arg(&mut out, 'o', &o.option),
|
|
CompleteOptionType::DoubleLong => append_switch_short_arg(&mut out, 'l', &o.option),
|
|
}
|
|
|
|
append_switch_short_arg(&mut out, 'd', o.localized_desc());
|
|
append_switch_short_arg(&mut out, 'a', &o.comp);
|
|
for c in &o.conditions {
|
|
append_switch_short_arg(&mut out, 'n', c);
|
|
}
|
|
out.push('\n');
|
|
|
|
out
|
|
}
|
|
|
|
/// Load command-specific completions for the specified command.
|
|
/// Returns `true` if something new was loaded, `false` if not.
|
|
pub fn complete_load(cmd: &wstr, parser: &Parser) -> bool {
|
|
let mut loaded_new = false;
|
|
|
|
// We have to load this as a function, since it may define a --wraps or signature.
|
|
// See issue #2466.
|
|
if function::load(cmd, parser) {
|
|
// We autoloaded something; check if we have a --wraps.
|
|
loaded_new |= !complete_get_wrap_targets(cmd).is_empty();
|
|
}
|
|
|
|
// It's important to NOT hold the lock around completion loading.
|
|
// We need to take the lock to decide what to load, drop it to perform the load, then reacquire
|
|
// it.
|
|
// Note we only look at the global fish_function_path and fish_complete_path.
|
|
let path_to_load = completion_autoloader
|
|
.lock()
|
|
.expect("mutex poisoned")
|
|
.resolve_command(cmd, &**EnvStack::globals());
|
|
if let Some(path_to_load) = path_to_load {
|
|
Autoload::perform_autoload(&path_to_load, parser);
|
|
completion_autoloader
|
|
.lock()
|
|
.expect("mutex poisoned")
|
|
.mark_autoload_finished(cmd);
|
|
loaded_new = true;
|
|
}
|
|
loaded_new
|
|
}
|
|
|
|
/// Return a list of all current completions.
|
|
/// Used by the bare `complete`, loaded completions are printed out as commands
|
|
pub fn complete_print(cmd: &wstr) -> WString {
|
|
let mut out = WString::new();
|
|
|
|
// Get references to our completions and sort them by order.
|
|
let completions = COMPLETION_MAP.lock().expect("poisoned mutex");
|
|
let mut completion_refs: Vec<_> = completions.iter().collect();
|
|
completion_refs.sort_by_key(|(_, c)| c.order);
|
|
|
|
for (key, entry) in completion_refs {
|
|
if !cmd.is_empty() && key.name != cmd {
|
|
continue;
|
|
}
|
|
|
|
// Output in reverse order to preserve legacy behavior (see #9221).
|
|
for o in entry.get_options().iter().rev() {
|
|
out.push_utfstr(&completion2string(key, o));
|
|
}
|
|
}
|
|
|
|
// Append wraps.
|
|
let wrappers = wrapper_map.lock().expect("poisoned mutex");
|
|
for (src, targets) in wrappers.iter() {
|
|
if !cmd.is_empty() && src != cmd {
|
|
continue;
|
|
}
|
|
for target in targets {
|
|
out.push_utfstr(L!("complete "));
|
|
out.push_utfstr(&escape(src));
|
|
append_switch_long_arg(&mut out, L!("wraps"), target);
|
|
out.push_utfstr(L!("\n"));
|
|
}
|
|
}
|
|
|
|
out
|
|
}
|
|
|
|
/// Observes that fish_complete_path has changed.
|
|
pub fn complete_invalidate_path() {
|
|
// TODO: here we unload all completions for commands that are loaded by the autoloader. We also
|
|
// unload any completions that the user may specified on the command line. We should in
|
|
// principle track those completions loaded by the autoloader alone.
|
|
|
|
let cmds = completion_autoloader
|
|
.lock()
|
|
.expect("mutex poisoned")
|
|
.get_autoloaded_commands();
|
|
for cmd in cmds {
|
|
complete_remove_all(cmd, false /* not a path */);
|
|
}
|
|
}
|
|
|
|
/// Adds a "wrap target." A wrap target is a command that completes like another command.
|
|
pub fn complete_add_wrapper(command: WString, new_target: WString) -> bool {
|
|
if command.is_empty() || new_target.is_empty() {
|
|
return false;
|
|
}
|
|
|
|
// If the command and the target are the same,
|
|
// there's no point in following the wrap-chain because we'd only complete the same thing.
|
|
// TODO: This should maybe include full cycle detection.
|
|
if command == new_target {
|
|
return false;
|
|
}
|
|
|
|
let mut wrappers = wrapper_map.lock().expect("poisoned mutex");
|
|
let targets = wrappers.entry(command).or_default();
|
|
// If it's already present, we do nothing.
|
|
if !targets.contains(&new_target) {
|
|
targets.push(new_target);
|
|
}
|
|
|
|
true
|
|
}
|
|
|
|
/// Removes a wrap target.
|
|
pub fn complete_remove_wrapper(command: WString, target_to_remove: &wstr) -> bool {
|
|
if command.is_empty() || target_to_remove.is_empty() {
|
|
return false;
|
|
}
|
|
|
|
let mut wrappers = wrapper_map.lock().expect("poisoned mutex");
|
|
let mut result = false;
|
|
for targets in wrappers.values_mut() {
|
|
if let Some(pos) = targets.iter().position(|t| t == target_to_remove) {
|
|
targets.remove(pos);
|
|
result = true;
|
|
}
|
|
}
|
|
|
|
result
|
|
}
|
|
|
|
/// Returns a list of wrap targets for a given command.
|
|
pub fn complete_get_wrap_targets(command: &wstr) -> Vec<WString> {
|
|
if command.is_empty() {
|
|
return vec![];
|
|
}
|
|
|
|
let wrappers = wrapper_map.lock().expect("poisoned mutex");
|
|
wrappers.get(command).cloned().unwrap_or_default()
|
|
}
|
|
|
|
pub struct CompletionListFfi(pub CompletionList);
|
|
|
|
pub use complete_ffi::CompletionRequestOptions;
|
|
|
|
#[cxx::bridge]
|
|
mod complete_ffi {
|
|
extern "C++" {
|
|
include!("complete.h");
|
|
include!("parser.h");
|
|
type Parser = crate::parser::Parser;
|
|
type OperationContext<'a> = crate::operation_context::OperationContext<'a>;
|
|
type wcstring_list_ffi_t = crate::ffi::wcstring_list_ffi_t;
|
|
}
|
|
extern "Rust" {
|
|
type Completion;
|
|
type CompletionListFfi;
|
|
|
|
fn new_completion() -> Box<Completion>;
|
|
fn new_completion_with(
|
|
completion: &CxxWString,
|
|
description: &CxxWString,
|
|
flags: u8,
|
|
) -> Box<Completion>;
|
|
fn completion(self: &Completion) -> UniquePtr<CxxWString>;
|
|
fn description(self: &Completion) -> UniquePtr<CxxWString>;
|
|
fn flags(self: &Completion) -> u8;
|
|
fn set_flags(self: &mut Completion, value: u8);
|
|
fn replaces_commandline(self: &Completion) -> bool;
|
|
fn match_is_exact_or_prefix(self: &Completion) -> bool;
|
|
fn completion_erase(self: &mut Completion, begin: usize, end: usize);
|
|
fn rank(self: &Completion) -> u32;
|
|
#[cxx_name = "clone"]
|
|
fn clone_ffi(self: &Completion) -> Box<Completion>;
|
|
|
|
fn new_completion_list() -> Box<CompletionListFfi>;
|
|
fn size(self: &CompletionListFfi) -> usize;
|
|
fn empty(self: &CompletionListFfi) -> bool;
|
|
fn at(self: &CompletionListFfi, i: usize) -> &Completion;
|
|
fn at_mut(self: &mut CompletionListFfi, i: usize) -> &mut Completion;
|
|
fn clear(self: &mut CompletionListFfi);
|
|
fn complete_invalidate_path();
|
|
fn reverse(self: &mut CompletionListFfi);
|
|
fn push_back(self: &mut CompletionListFfi, completion: &Completion);
|
|
fn sort_and_prioritize(self: &mut CompletionListFfi, flags: CompletionRequestOptions);
|
|
#[cxx_name = "complete_load"]
|
|
fn complete_load_ffi(cmd: &CxxWString, parser: &Parser) -> bool;
|
|
#[cxx_name = "complete"]
|
|
fn complete_ffi(
|
|
search_string: &CxxWString,
|
|
complete_flags: CompletionRequestOptions,
|
|
ctx: &OperationContext<'static>,
|
|
needs_load: &mut UniquePtr<wcstring_list_ffi_t>,
|
|
) -> Box<CompletionListFfi>;
|
|
#[cxx_name = "append_completion"]
|
|
fn append_completion_ffi(completions: Pin<&mut CompletionListFfi>, comp: &CxxWString);
|
|
}
|
|
|
|
#[derive(Clone, Copy)]
|
|
pub struct CompletionRequestOptions {
|
|
/// Requesting autosuggestion
|
|
pub autosuggestion: bool,
|
|
/// Make descriptions
|
|
pub descriptions: bool,
|
|
/// If set, we do not require a prefix match
|
|
pub fuzzy_match: bool,
|
|
}
|
|
|
|
extern "Rust" {
|
|
fn completion_request_options_autosuggest() -> CompletionRequestOptions;
|
|
fn completion_request_options_normal() -> CompletionRequestOptions;
|
|
}
|
|
}
|
|
|
|
fn complete_ffi(
|
|
search_string: &CxxWString,
|
|
complete_flags: CompletionRequestOptions,
|
|
ctx: &OperationContext<'static>,
|
|
needs_load: &mut UniquePtr<crate::ffi::wcstring_list_ffi_t>,
|
|
) -> Box<CompletionListFfi> {
|
|
let (completions, to_load) = complete(search_string.as_wstr(), complete_flags, ctx);
|
|
if !needs_load.is_null() {
|
|
*needs_load = to_load.to_ffi();
|
|
}
|
|
Box::new(CompletionListFfi(completions))
|
|
}
|
|
|
|
fn completion_request_options_autosuggest() -> CompletionRequestOptions {
|
|
CompletionRequestOptions::autosuggest()
|
|
}
|
|
fn completion_request_options_normal() -> CompletionRequestOptions {
|
|
CompletionRequestOptions::normal()
|
|
}
|
|
|
|
impl Default for CompletionRequestOptions {
|
|
fn default() -> Self {
|
|
Self {
|
|
autosuggestion: false,
|
|
descriptions: false,
|
|
fuzzy_match: false,
|
|
}
|
|
}
|
|
}
|
|
|
|
unsafe impl cxx::ExternType for CompletionListFfi {
|
|
type Id = cxx::type_id!("CompletionListFfi");
|
|
type Kind = cxx::kind::Opaque;
|
|
}
|
|
|
|
fn new_completion() -> Box<Completion> {
|
|
Box::new(Completion::new(
|
|
"".into(),
|
|
"".into(),
|
|
StringFuzzyMatch::exact_match(),
|
|
CompleteFlags::default(),
|
|
))
|
|
}
|
|
fn new_completion_with(
|
|
completion: &CxxWString,
|
|
description: &CxxWString,
|
|
flags: u8,
|
|
) -> Box<Completion> {
|
|
Box::new(Completion::new(
|
|
completion.from_ffi(),
|
|
description.from_ffi(),
|
|
StringFuzzyMatch::exact_match(),
|
|
CompleteFlags::from_bits(flags).unwrap(),
|
|
))
|
|
}
|
|
fn new_completion_list() -> Box<CompletionListFfi> {
|
|
Box::new(CompletionListFfi(CompletionList::new()))
|
|
}
|
|
fn append_completion_ffi(completions: Pin<&mut CompletionListFfi>, comp: &CxxWString) {
|
|
completions.get_mut().0.push(Completion::new(
|
|
comp.from_ffi(),
|
|
"".into(),
|
|
StringFuzzyMatch::exact_match(),
|
|
CompleteFlags::default(),
|
|
));
|
|
}
|
|
|
|
impl Completion {
|
|
fn completion(&self) -> UniquePtr<CxxWString> {
|
|
self.completion.to_ffi()
|
|
}
|
|
fn description(&self) -> UniquePtr<CxxWString> {
|
|
self.description.to_ffi()
|
|
}
|
|
fn flags(&self) -> u8 {
|
|
self.flags.bits()
|
|
}
|
|
fn set_flags(&mut self, value: u8) {
|
|
self.flags = CompleteFlags::from_bits(value).unwrap();
|
|
}
|
|
fn clone_ffi(&self) -> Box<Completion> {
|
|
Box::new(self.clone())
|
|
}
|
|
fn match_is_exact_or_prefix(&self) -> bool {
|
|
self.r#match.is_exact_or_prefix()
|
|
}
|
|
fn completion_erase(&mut self, begin: usize, end: usize) {
|
|
self.completion.replace_range(begin..end, L!(""))
|
|
}
|
|
}
|
|
impl CompletionListFfi {
|
|
fn size(&self) -> usize {
|
|
self.0.len()
|
|
}
|
|
fn empty(&self) -> bool {
|
|
self.0.is_empty()
|
|
}
|
|
fn at(&self, i: usize) -> &Completion {
|
|
&self.0[i]
|
|
}
|
|
fn at_mut(&mut self, i: usize) -> &mut Completion {
|
|
&mut self.0[i]
|
|
}
|
|
fn reverse(&mut self) {
|
|
self.0.reverse();
|
|
}
|
|
fn clear(&mut self) {
|
|
self.0.clear();
|
|
}
|
|
fn push_back(&mut self, completion: &Completion) {
|
|
self.0.push(completion.clone());
|
|
}
|
|
fn sort_and_prioritize(&mut self, flags: CompletionRequestOptions) {
|
|
sort_and_prioritize(&mut self.0, flags);
|
|
}
|
|
}
|
|
|
|
fn complete_load_ffi(cmd: &CxxWString, parser: &Parser) -> bool {
|
|
complete_load(cmd.as_wstr(), parser)
|
|
}
|