mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-19 08:24:00 +00:00
Simplify history searching and fix deduplication
The history search logic had a not very useful "fast path" which was also buggy because it neglected to dedup. Switch the "fast path" to just a history search type which always matches. Fixes #6278
This commit is contained in:
parent
52e900690b
commit
47c0b5f931
2 changed files with 46 additions and 61 deletions
|
@ -206,6 +206,9 @@ bool history_item_t::matches_search(const wcstring &term, enum history_search_ty
|
||||||
if (wcpattern2.back() != ANY_STRING) wcpattern2.push_back(ANY_STRING);
|
if (wcpattern2.back() != ANY_STRING) wcpattern2.push_back(ANY_STRING);
|
||||||
return wildcard_match(content_to_match, wcpattern2);
|
return wildcard_match(content_to_match, wcpattern2);
|
||||||
}
|
}
|
||||||
|
case history_search_type_t::match_everything: {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
DIE("unexpected history_search_type_t value");
|
DIE("unexpected history_search_type_t value");
|
||||||
}
|
}
|
||||||
|
@ -1343,76 +1346,60 @@ void history_t::resolve_pending() { impl()->resolve_pending(); }
|
||||||
|
|
||||||
void history_t::save() { impl()->save(); }
|
void history_t::save() { impl()->save(); }
|
||||||
|
|
||||||
|
/// Perform a search of \p hist for \p search_string. Invoke a function \p func for each match. If
|
||||||
|
/// \p func returns true, continue the search; else stop it.
|
||||||
|
static void do_1_history_search(history_t &hist, history_search_type_t search_type,
|
||||||
|
const wcstring &search_string, bool case_sensitive,
|
||||||
|
const std::function<bool(const history_item_t &item)> &func) {
|
||||||
|
history_search_t searcher = history_search_t(hist, search_string, search_type,
|
||||||
|
case_sensitive ? 0 : history_search_ignore_case);
|
||||||
|
while (searcher.go_backwards()) {
|
||||||
|
if (!func(searcher.current_item())) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Searches history.
|
// Searches history.
|
||||||
bool history_t::search(history_search_type_t search_type, const wcstring_list_t &search_args,
|
bool history_t::search(history_search_type_t search_type, const wcstring_list_t &search_args,
|
||||||
const wchar_t *show_time_format, size_t max_items, bool case_sensitive,
|
const wchar_t *show_time_format, size_t max_items, bool case_sensitive,
|
||||||
bool null_terminate, bool reverse, io_streams_t &streams) {
|
bool null_terminate, bool reverse, io_streams_t &streams) {
|
||||||
if (!search_args.empty()) {
|
wcstring_list_t collected;
|
||||||
// User wants the results filtered. This is not the common case so we do it separate
|
|
||||||
// from the code below for unfiltered output which is much cheaper.
|
|
||||||
return search_with_args(search_type, search_args, show_time_format, max_items,
|
|
||||||
case_sensitive, null_terminate, reverse, streams);
|
|
||||||
}
|
|
||||||
|
|
||||||
// scoped_lock locker(lock);
|
|
||||||
size_t hist_size = this->size();
|
|
||||||
if (max_items > hist_size) max_items = hist_size;
|
|
||||||
|
|
||||||
wcstring formatted_record;
|
wcstring formatted_record;
|
||||||
|
size_t remaining = max_items;
|
||||||
|
|
||||||
|
// The function we use to act on each item.
|
||||||
|
std::function<bool(const history_item_t &item)> func = [&](const history_item_t &item) -> bool {
|
||||||
|
if (remaining == 0) return false;
|
||||||
|
remaining -= 1;
|
||||||
|
format_history_record(item, show_time_format, null_terminate, &formatted_record);
|
||||||
if (reverse) {
|
if (reverse) {
|
||||||
for (size_t i = max_items; i != 0; --i) {
|
// We need to collect this for later.
|
||||||
auto cur_item = this->item_at_index(i);
|
collected.push_back(std::move(formatted_record));
|
||||||
format_history_record(cur_item, show_time_format, null_terminate, &formatted_record);
|
|
||||||
streams.out.append(formatted_record);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Start at one because zero is the current command.
|
// We can output this immediately.
|
||||||
for (size_t i = 1; i < max_items + 1; ++i) {
|
|
||||||
auto cur_item = this->item_at_index(i);
|
|
||||||
wcstring result;
|
|
||||||
format_history_record(cur_item, show_time_format, null_terminate, &formatted_record);
|
|
||||||
streams.out.append(formatted_record);
|
streams.out.append(formatted_record);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
};
|
||||||
|
|
||||||
bool history_t::search_with_args(history_search_type_t search_type,
|
|
||||||
const wcstring_list_t &search_args,
|
|
||||||
const wchar_t *show_time_format, size_t max_items,
|
|
||||||
bool case_sensitive, bool null_terminate, bool reverse,
|
|
||||||
io_streams_t &streams) {
|
|
||||||
wcstring_list_t results;
|
|
||||||
size_t hist_size = this->size();
|
|
||||||
if (max_items > hist_size) max_items = hist_size;
|
|
||||||
|
|
||||||
|
if (search_args.empty()) {
|
||||||
|
// The user had no search terms; just append everything.
|
||||||
|
do_1_history_search(*this, history_search_type_t::match_everything, {}, false, func);
|
||||||
|
} else {
|
||||||
for (const wcstring &search_string : search_args) {
|
for (const wcstring &search_string : search_args) {
|
||||||
if (search_string.empty()) {
|
if (search_string.empty()) {
|
||||||
streams.err.append_format(L"Searching for the empty string isn't allowed");
|
streams.err.append_format(L"Searching for the empty string isn't allowed");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
history_search_t searcher = history_search_t(
|
do_1_history_search(*this, search_type, search_string, case_sensitive, func);
|
||||||
*this, search_string, search_type, case_sensitive ? 0 : history_search_ignore_case);
|
|
||||||
wcstring formatted_record;
|
|
||||||
while (searcher.go_backwards()) {
|
|
||||||
const auto &cur_item = searcher.current_item();
|
|
||||||
format_history_record(cur_item, show_time_format, null_terminate, &formatted_record);
|
|
||||||
if (reverse) {
|
|
||||||
results.push_back(formatted_record);
|
|
||||||
} else {
|
|
||||||
streams.out.append(formatted_record);
|
|
||||||
}
|
|
||||||
if (--max_items == 0) break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (reverse) {
|
// Output any items we collected (which only happens in reverse).
|
||||||
for (auto it = results.rbegin(); it != results.rend(); it++) {
|
for (auto iter = collected.rbegin(); iter != collected.rend(); ++iter) {
|
||||||
streams.out.append(*it);
|
streams.out.append(*iter);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -55,6 +55,8 @@ enum class history_search_type_t {
|
||||||
contains_glob,
|
contains_glob,
|
||||||
// Search for commands starting with the given glob pattern.
|
// Search for commands starting with the given glob pattern.
|
||||||
prefix_glob,
|
prefix_glob,
|
||||||
|
// Matches everything.
|
||||||
|
match_everything,
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef uint64_t history_identifier_t;
|
typedef uint64_t history_identifier_t;
|
||||||
|
@ -172,10 +174,6 @@ class history_t {
|
||||||
const wchar_t *show_time_format, size_t max_items, bool case_sensitive,
|
const wchar_t *show_time_format, size_t max_items, bool case_sensitive,
|
||||||
bool null_terminate, bool reverse, io_streams_t &streams);
|
bool null_terminate, bool reverse, io_streams_t &streams);
|
||||||
|
|
||||||
bool search_with_args(history_search_type_t search_type, const wcstring_list_t &search_args,
|
|
||||||
const wchar_t *show_time_format, size_t max_items, bool case_sensitive,
|
|
||||||
bool null_terminate, bool reverse, io_streams_t &streams);
|
|
||||||
|
|
||||||
// Irreversibly clears history.
|
// Irreversibly clears history.
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue