Introduce completion_receiver_t

completion_receiver_t wraps a completion list; it will centralize logic
around adding completions and most importantly it will enforce that we
do not exceed our expansion limit.
This commit is contained in:
ridiculousfish 2020-12-01 11:36:38 -08:00
parent a614a19b07
commit af3383e727
4 changed files with 97 additions and 37 deletions

View file

@ -255,9 +255,13 @@ bool contains(const Col &col, const T2 &val) {
/// Append a vector \p donator to the vector \p receiver. /// Append a vector \p donator to the vector \p receiver.
template <typename T> template <typename T>
void vec_append(std::vector<T> &receiver, std::vector<T> &&donator) { void vec_append(std::vector<T> &receiver, std::vector<T> &&donator) {
if (receiver.empty()) {
receiver = std::move(donator);
} else {
receiver.insert(receiver.end(), std::make_move_iterator(donator.begin()), receiver.insert(receiver.end(), std::make_move_iterator(donator.begin()),
std::make_move_iterator(donator.end())); std::make_move_iterator(donator.end()));
} }
}
/// Move an object into a shared_ptr. /// Move an object into a shared_ptr.
template <typename T> template <typename T>

View file

@ -236,6 +236,32 @@ void completion_t::prepend_token_prefix(const wcstring &prefix) {
} }
} }
void completion_receiver_t::add(completion_t &&comp) {
this->completions_.push_back(std::move(comp));
}
void completion_receiver_t::add(wcstring &&comp) { this->add(std::move(comp), wcstring{}); }
void completion_receiver_t::add(wcstring &&comp, wcstring &&desc, complete_flags_t flags,
string_fuzzy_match_t match) {
this->completions_.emplace_back(std::move(comp), std::move(desc), match, flags);
}
void completion_receiver_t::add_list(completion_list_t &&lst) {
if (completions_.empty()) {
completions_ = std::move(lst);
} else {
completions_.reserve(completions_.size() + lst.size());
std::move(lst.begin(), lst.end(), std::back_inserter(completions_));
}
}
completion_list_t completion_receiver_t::take() {
completion_list_t res{};
std::swap(res, this->completions_);
return res;
}
// If these functions aren't force inlined, it is actually faster to call // 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! // stable_sort twice rather than to iterate once performing all comparisons in one go!
__attribute__((always_inline)) static inline bool compare_completions_by_duplicate_arguments( __attribute__((always_inline)) static inline bool compare_completions_by_duplicate_arguments(

View file

@ -120,6 +120,37 @@ using completion_request_flags_t = enum_set_t<completion_request_t>;
class completion_t; class completion_t;
using completion_list_t = std::vector<completion_t>; using completion_list_t = std::vector<completion_t>;
/// A completion receiver accepts completions. It is essentially a wrapper around std::vector with
/// some conveniences.
class completion_receiver_t {
public:
/// Add a completion.
void add(completion_t &&comp);
/// Add a completion with the given string, and default other properties.
void add(wcstring &&comp);
/// Add a completion with the given string, description, flags, and fuzzy match.
void add(wcstring &&comp, wcstring &&desc, complete_flags_t flags = 0,
string_fuzzy_match_t match = string_fuzzy_match_t::exact_match());
/// Add a list of completions.
void add_list(completion_list_t &&lst);
/// Swap our completions with a new list.
void swap(completion_list_t &lst) { std::swap(completions_, lst); }
/// Clear the list of completions. This retains the storage inside completions_ which can be
/// useful to prevent allocations.
void clear() { completions_.clear(); }
/// \return the list of completions, clearing them.
completion_list_t take();
private:
completion_list_t completions_;
};
enum complete_option_type_t { enum complete_option_type_t {
option_type_args_only, // no option option_type_args_only, // no option
option_type_short, // -x option_type_short, // -x

View file

@ -285,14 +285,14 @@ static size_t parse_slice(const wchar_t *in, wchar_t **end_ptr, std::vector<long
/// Note: last_idx is considered to be where it previously finished procesisng. This means it /// Note: last_idx is considered to be where it previously finished procesisng. This means it
/// actually starts operating on last_idx-1. As such, to process a string fully, pass string.size() /// actually starts operating on last_idx-1. As such, to process a string fully, pass string.size()
/// as last_idx instead of string.size()-1. /// as last_idx instead of string.size()-1.
static bool expand_variables(wcstring instr, completion_list_t *out, size_t last_idx, static bool expand_variables(wcstring instr, completion_receiver_t *out, size_t last_idx,
const environment_t &vars, parse_error_list_t *errors) { const environment_t &vars, parse_error_list_t *errors) {
const size_t insize = instr.size(); const size_t insize = instr.size();
// last_idx may be 1 past the end of the string, but no further. // last_idx may be 1 past the end of the string, but no further.
assert(last_idx <= insize && "Invalid last_idx"); assert(last_idx <= insize && "Invalid last_idx");
if (last_idx == 0) { if (last_idx == 0) {
append_completion(out, std::move(instr)); out->add(std::move(instr));
return true; return true;
} }
@ -308,7 +308,7 @@ static bool expand_variables(wcstring instr, completion_list_t *out, size_t last
} }
if (varexp_char_idx >= instr.size()) { if (varexp_char_idx >= instr.size()) {
// No variable expand char, we're done. // No variable expand char, we're done.
append_completion(out, std::move(instr)); out->add(std::move(instr));
return true; return true;
} }
@ -458,9 +458,9 @@ static bool expand_variables(wcstring instr, completion_list_t *out, size_t last
return expand_variables(std::move(res), out, varexp_char_idx, vars, errors); return expand_variables(std::move(res), out, varexp_char_idx, vars, errors);
} else { } else {
// Normal cartesian-product expansion. // Normal cartesian-product expansion.
for (const wcstring &item : var_item_list) { for (wcstring &item : var_item_list) {
if (varexp_char_idx == 0 && var_name_and_slice_stop == insize) { if (varexp_char_idx == 0 && var_name_and_slice_stop == insize) {
append_completion(out, item); out->add(std::move(item));
} else { } else {
wcstring new_in(instr, 0, varexp_char_idx); wcstring new_in(instr, 0, varexp_char_idx);
if (!new_in.empty()) { if (!new_in.empty()) {
@ -482,8 +482,8 @@ static bool expand_variables(wcstring instr, completion_list_t *out, size_t last
} }
/// Perform brace expansion, placing the expanded strings into \p out. /// Perform brace expansion, placing the expanded strings into \p out.
static expand_result_t expand_braces(wcstring &&instr, expand_flags_t flags, completion_list_t *out, static expand_result_t expand_braces(wcstring &&instr, expand_flags_t flags,
parse_error_list_t *errors) { completion_receiver_t *out, parse_error_list_t *errors) {
bool syntax_error = false; bool syntax_error = false;
int brace_count = 0; int brace_count = 0;
@ -549,7 +549,7 @@ static expand_result_t expand_braces(wcstring &&instr, expand_flags_t flags, com
} }
if (brace_begin == nullptr) { if (brace_begin == nullptr) {
append_completion(out, std::move(instr)); out->add(std::move(instr));
return expand_result_t::ok; return expand_result_t::ok;
} }
@ -594,7 +594,7 @@ static expand_result_t expand_braces(wcstring &&instr, expand_flags_t flags, com
/// Expand a command substitution \p input, executing on \p ctx, and inserting the results into /// Expand a command substitution \p input, executing on \p ctx, and inserting the results into
/// \p out_list, or any errors into \p errors. \return an expand result. /// \p out_list, or any errors into \p errors. \return an expand result.
static expand_result_t expand_cmdsubst(wcstring input, const operation_context_t &ctx, static expand_result_t expand_cmdsubst(wcstring input, const operation_context_t &ctx,
completion_list_t *out_list, parse_error_list_t *errors) { completion_receiver_t *out, parse_error_list_t *errors) {
assert(ctx.parser && "Cannot expand without a parser"); assert(ctx.parser && "Cannot expand without a parser");
size_t cursor = 0; size_t cursor = 0;
size_t paren_begin = 0; size_t paren_begin = 0;
@ -608,7 +608,7 @@ static expand_result_t expand_cmdsubst(wcstring input, const operation_context_t
return expand_result_t::make_error(STATUS_EXPAND_ERROR); return expand_result_t::make_error(STATUS_EXPAND_ERROR);
} }
case 0: { case 0: {
append_completion(out_list, std::move(input)); out->add(std::move(input));
return expand_result_t::ok; return expand_result_t::ok;
} }
case 1: { case 1: {
@ -671,9 +671,10 @@ static expand_result_t expand_cmdsubst(wcstring input, const operation_context_t
// Recursively call ourselves to expand any remaining command substitutions. The result of this // Recursively call ourselves to expand any remaining command substitutions. The result of this
// recursive call using the tail of the string is inserted into the tail_expand array list // recursive call using the tail of the string is inserted into the tail_expand array list
completion_list_t tail_expand; completion_receiver_t tail_expand_recv;
expand_cmdsubst(input.substr(tail_begin), ctx, &tail_expand, expand_cmdsubst(input.substr(tail_begin), ctx, &tail_expand_recv,
errors); // TODO: offset error locations errors); // TODO: offset error locations
completion_list_t tail_expand = tail_expand_recv.take();
// Combine the result of the current command substitution with the result of the recursive tail // Combine the result of the current command substitution with the result of the recursive tail
// expansion. // expansion.
@ -688,7 +689,7 @@ static expand_result_t expand_cmdsubst(wcstring input, const operation_context_t
whole_item.append(sub_item2); whole_item.append(sub_item2);
whole_item.push_back(INTERNAL_SEPARATOR); whole_item.push_back(INTERNAL_SEPARATOR);
whole_item.append(tail_item.completion); whole_item.append(tail_item.completion);
append_completion(out_list, std::move(whole_item)); out->add(std::move(whole_item));
} }
} }
@ -870,13 +871,13 @@ class expander_t {
/// An expansion stage is a member function pointer. /// An expansion stage is a member function pointer.
/// It accepts the input string (transferring ownership) and returns the list of output /// It accepts the input string (transferring ownership) and returns the list of output
/// completions by reference. It may return an error, which halts expansion. /// completions by reference. It may return an error, which halts expansion.
using stage_t = expand_result_t (expander_t::*)(wcstring, completion_list_t *); using stage_t = expand_result_t (expander_t::*)(wcstring, completion_receiver_t *);
expand_result_t stage_cmdsubst(wcstring input, completion_list_t *out); expand_result_t stage_cmdsubst(wcstring input, completion_receiver_t *out);
expand_result_t stage_variables(wcstring input, completion_list_t *out); expand_result_t stage_variables(wcstring input, completion_receiver_t *out);
expand_result_t stage_braces(wcstring input, completion_list_t *out); expand_result_t stage_braces(wcstring input, completion_receiver_t *out);
expand_result_t stage_home_and_self(wcstring input, completion_list_t *out); expand_result_t stage_home_and_self(wcstring input, completion_receiver_t *out);
expand_result_t stage_wildcards(wcstring path_to_expand, completion_list_t *out); expand_result_t stage_wildcards(wcstring path_to_expand, completion_receiver_t *out);
expander_t(const operation_context_t &ctx, expand_flags_t flags, parse_error_list_t *errors) expander_t(const operation_context_t &ctx, expand_flags_t flags, parse_error_list_t *errors)
: ctx(ctx), flags(flags), errors(errors) {} : ctx(ctx), flags(flags), errors(errors) {}
@ -887,12 +888,12 @@ class expander_t {
parse_error_list_t *errors); parse_error_list_t *errors);
}; };
expand_result_t expander_t::stage_cmdsubst(wcstring input, completion_list_t *out) { expand_result_t expander_t::stage_cmdsubst(wcstring input, completion_receiver_t *out) {
if (flags & expand_flag::skip_cmdsubst) { if (flags & expand_flag::skip_cmdsubst) {
size_t cur = 0, start = 0, end; size_t cur = 0, start = 0, end;
switch (parse_util_locate_cmdsubst_range(input, &cur, nullptr, &start, &end, true)) { switch (parse_util_locate_cmdsubst_range(input, &cur, nullptr, &start, &end, true)) {
case 0: case 0:
append_completion(out, std::move(input)); out->add(std::move(input));
return expand_result_t::ok; return expand_result_t::ok;
case 1: case 1:
append_cmdsub_error(errors, start, L"Command substitutions not allowed"); append_cmdsub_error(errors, start, L"Command substitutions not allowed");
@ -907,7 +908,7 @@ expand_result_t expander_t::stage_cmdsubst(wcstring input, completion_list_t *ou
} }
} }
expand_result_t expander_t::stage_variables(wcstring input, completion_list_t *out) { expand_result_t expander_t::stage_variables(wcstring input, completion_receiver_t *out) {
// We accept incomplete strings here, since complete uses expand_string to expand incomplete // We accept incomplete strings here, since complete uses expand_string to expand incomplete
// strings from the commandline. // strings from the commandline.
wcstring next; wcstring next;
@ -919,7 +920,7 @@ expand_result_t expander_t::stage_variables(wcstring input, completion_list_t *o
i = L'$'; i = L'$';
} }
} }
append_completion(out, std::move(next)); out->add(std::move(next));
} else { } else {
size_t size = next.size(); size_t size = next.size();
if (!expand_variables(std::move(next), out, size, ctx.vars, errors)) { if (!expand_variables(std::move(next), out, size, ctx.vars, errors)) {
@ -929,20 +930,20 @@ expand_result_t expander_t::stage_variables(wcstring input, completion_list_t *o
return expand_result_t::ok; return expand_result_t::ok;
} }
expand_result_t expander_t::stage_braces(wcstring input, completion_list_t *out) { expand_result_t expander_t::stage_braces(wcstring input, completion_receiver_t *out) {
return expand_braces(std::move(input), flags, out, errors); return expand_braces(std::move(input), flags, out, errors);
} }
expand_result_t expander_t::stage_home_and_self(wcstring input, completion_list_t *out) { expand_result_t expander_t::stage_home_and_self(wcstring input, completion_receiver_t *out) {
if (!(flags & expand_flag::skip_home_directories)) { if (!(flags & expand_flag::skip_home_directories)) {
expand_home_directory(input, ctx.vars); expand_home_directory(input, ctx.vars);
} }
expand_percent_self(input); expand_percent_self(input);
append_completion(out, std::move(input)); out->add(std::move(input));
return expand_result_t::ok; return expand_result_t::ok;
} }
expand_result_t expander_t::stage_wildcards(wcstring path_to_expand, completion_list_t *out) { expand_result_t expander_t::stage_wildcards(wcstring path_to_expand, completion_receiver_t *out) {
expand_result_t result = expand_result_t::ok; expand_result_t result = expand_result_t::ok;
remove_internal_separator(&path_to_expand, flags & expand_flag::skip_wildcards); remove_internal_separator(&path_to_expand, flags & expand_flag::skip_wildcards);
@ -1026,14 +1027,13 @@ expand_result_t expander_t::stage_wildcards(wcstring path_to_expand, completion_
[&](const completion_t &a, const completion_t &b) { [&](const completion_t &a, const completion_t &b) {
return wcsfilecmp_glob(a.completion.c_str(), b.completion.c_str()) < 0; return wcsfilecmp_glob(a.completion.c_str(), b.completion.c_str()) < 0;
}); });
out->add_list(std::move(expanded));
std::move(expanded.begin(), expanded.end(), std::back_inserter(*out));
} else { } else {
// Can't fully justify this check. I think it's that SKIP_WILDCARDS is used when completing // Can't fully justify this check. I think it's that SKIP_WILDCARDS is used when completing
// to mean don't do file expansions, so if we're not doing file expansions, just drop this // to mean don't do file expansions, so if we're not doing file expansions, just drop this
// completion on the floor. // completion on the floor.
if (!(flags & expand_flag::for_completions)) { if (!(flags & expand_flag::for_completions)) {
append_completion(out, std::move(path_to_expand)); out->add(std::move(path_to_expand));
} }
} }
return result; return result;
@ -1058,9 +1058,10 @@ expand_result_t expander_t::expand_string(wcstring input, completion_list_t *out
&expander_t::stage_wildcards}; &expander_t::stage_wildcards};
// Load up our single initial completion. // Load up our single initial completion.
completion_list_t completions, output_storage; completion_list_t completions;
append_completion(&completions, input); append_completion(&completions, input);
completion_receiver_t output_storage;
expand_result_t total_result = expand_result_t::ok; expand_result_t total_result = expand_result_t::ok;
for (stage_t stage : stages) { for (stage_t stage : stages) {
for (completion_t &comp : completions) { for (completion_t &comp : completions) {
@ -1077,7 +1078,7 @@ expand_result_t expander_t::expand_string(wcstring input, completion_list_t *out
} }
// Output becomes our next stage's input. // Output becomes our next stage's input.
completions.swap(output_storage); output_storage.swap(completions);
output_storage.clear(); output_storage.clear();
if (total_result == expand_result_t::error) { if (total_result == expand_result_t::error) {
break; break;
@ -1099,9 +1100,7 @@ expand_result_t expander_t::expand_string(wcstring input, completion_list_t *out
if (!(flags & expand_flag::skip_home_directories)) { if (!(flags & expand_flag::skip_home_directories)) {
unexpand_tildes(input, ctx.vars, &completions); unexpand_tildes(input, ctx.vars, &completions);
} }
out_completions->insert(out_completions->end(), vec_append(*out_completions, std::move(completions));
std::make_move_iterator(completions.begin()),
std::make_move_iterator(completions.end()));
} }
return total_result; return total_result;
} }