From aa1b065dd1c03f7c0867d002badcf0dc88feb3c4 Mon Sep 17 00:00:00 2001 From: ridiculousfish Date: Fri, 28 Mar 2014 23:22:03 -0700 Subject: [PATCH] Allow appending path hints to history items after they have been added, allowing us to avoid the delay before items appear in history. Should fix #984 --- fish_tests.cpp | 5 +- history.cpp | 189 ++++++++++++++++++++++++++++++++++++------------- history.h | 51 ++++++++----- reader.cpp | 2 +- 4 files changed, 178 insertions(+), 69 deletions(-) diff --git a/fish_tests.cpp b/fish_tests.cpp index 3cca2a0c0..92bd0f717 100644 --- a/fish_tests.cpp +++ b/fish_tests.cpp @@ -2203,7 +2203,7 @@ void history_tests_t::test_history(void) test_history_matches(search3, 0); /* Test history escaping and unescaping, yaml, etc. */ - std::vector before, after; + history_item_list_t before, after; history.clear(); size_t i, max = 100; for (i=1; i <= max; i++) @@ -2225,7 +2225,8 @@ void history_tests_t::test_history(void) } /* Record this item */ - history_item_t item(value, time(NULL), paths); + history_item_t item(value, time(NULL)); + item.required_paths = paths; before.push_back(item); history.add(item); } diff --git a/history.cpp b/history.cpp index f8183f5d8..de4056998 100644 --- a/history.cpp +++ b/history.cpp @@ -22,6 +22,7 @@ #include "sanity.h" #include "tokenizer.h" #include "reader.h" +#include "parse_tree.h" #include "wutil.h" #include "history.h" @@ -235,7 +236,7 @@ static void escape_yaml(std::string &str); /** Undoes escape_yaml */ static void unescape_yaml(std::string &str); -/* We can merge two items if they are the same command. We use the more recent timestamp and the longer list of required paths. */ +/* We can merge two items if they are the same command. We use the more recent timestamp, more recent identifier, and the longer list of required paths. */ bool history_item_t::merge(const history_item_t &item) { bool result = false; @@ -246,16 +247,20 @@ bool history_item_t::merge(const history_item_t &item) { this->required_paths = item.required_paths; } + if (this->identifier < item.identifier) + { + this->identifier = item.identifier; + } result = true; } return result; } -history_item_t::history_item_t(const wcstring &str) : contents(str), creation_timestamp(time(NULL)) +history_item_t::history_item_t(const wcstring &str) : contents(str), creation_timestamp(time(NULL)), identifier(0) { } -history_item_t::history_item_t(const wcstring &str, time_t when, const path_list_t &paths) : contents(str), creation_timestamp(when), required_paths(paths) +history_item_t::history_item_t(const wcstring &str, time_t when, history_identifier_t ident) : contents(str), creation_timestamp(when), identifier(ident) { } @@ -543,6 +548,7 @@ history_t & history_t::history_with_name(const wcstring &name) history_t::history_t(const wcstring &pname) : name(pname), first_unwritten_new_item_index(0), + disable_automatic_save_counter(0), mmap_start(NULL), mmap_length(0), mmap_file_id(kInvalidFileID), @@ -572,8 +578,21 @@ void history_t::add(const history_item_t &item) { /* We have to add a new item */ new_items.push_back(item); + save_internal_unless_disabled(); } +} +void history_t::save_internal_unless_disabled() +{ + /* This must be called while locked */ + ASSERT_IS_LOCKED(lock); + + /* Respect disable_automatic_save_counter */ + if (disable_automatic_save_counter > 0) + { + return; + } + /* We may or may not vacuum. We try to vacuum every kVacuumFrequency items, but start the countdown at a random number so that even if the user never runs more than 25 commands, we'll eventually vacuum. If countdown_to_vacuum is -1, it means we haven't yet picked a value for the counter. */ const int kVacuumFrequency = 25; if (countdown_to_vacuum < 0) @@ -582,7 +601,7 @@ void history_t::add(const history_item_t &item) /* Generate a number in the range [0, kVacuumFrequency) */ countdown_to_vacuum = rand_r(&seed) / (RAND_MAX / kVacuumFrequency + 1); } - + /* Determine if we're going to vacuum */ bool vacuum = false; if (countdown_to_vacuum == 0) @@ -590,18 +609,17 @@ void history_t::add(const history_item_t &item) countdown_to_vacuum = kVacuumFrequency; vacuum = true; } - + /* This might be a good candidate for moving to a background thread */ time_profiler_t profiler(vacuum ? "save_internal vacuum" : "save_internal no vacuum"); this->save_internal(vacuum); - + /* Update our countdown */ assert(countdown_to_vacuum > 0); countdown_to_vacuum--; - } -void history_t::add(const wcstring &str, const path_list_t &valid_paths) +void history_t::add(const wcstring &str, history_identifier_t ident) { time_t when = time(NULL); /* Big hack: do not allow timestamps equal to our birthdate. This is because we include items whose timestamps are equal to our birthdate when reading old history, so we can catch "just closed" items. But this means that we may interpret our own items, that we just wrote, as old items, if we wrote them in the same second as our birthdate. @@ -609,7 +627,7 @@ void history_t::add(const wcstring &str, const path_list_t &valid_paths) if (when == this->birth_timestamp) when++; - this->add(history_item_t(str, when, valid_paths)); + this->add(history_item_t(str, when, ident)); } void history_t::remove(const wcstring &str) @@ -621,7 +639,7 @@ void history_t::remove(const wcstring &str) size_t idx = new_items.size(); while (idx--) { - if (new_items[idx].str() == str) + if (new_items.at(idx).str() == str) { new_items.erase(new_items.begin() + idx); @@ -635,6 +653,28 @@ void history_t::remove(const wcstring &str) assert(first_unwritten_new_item_index <= new_items.size()); } +void history_t::set_valid_file_paths(const wcstring_list_t &valid_file_paths, history_identifier_t ident) +{ + /* 0 identifier is used to mean "not necessary" */ + if (ident == 0) + { + return; + } + + scoped_lock locker(lock); + + /* Look for an item with the given identifier. It is likely to be at the end of new_items */ + for (history_item_list_t::reverse_iterator iter = new_items.rbegin(); iter != new_items.rend(); iter++) + { + if (iter->identifier == ident) + { + /* Found it */ + iter->required_paths = valid_file_paths; + break; + } + } +} + void history_t::get_string_representation(wcstring &result, const wcstring &separator) { scoped_lock locker(lock); @@ -644,7 +684,7 @@ void history_t::get_string_representation(wcstring &result, const wcstring &sepa std::set seen; /* Append new items. Note that in principle we could use const_reverse_iterator, but we do not because reverse_iterator is not convertible to const_reverse_iterator ( http://github.com/fish-shell/fish-shell/issues/431 ) */ - for (std::vector::reverse_iterator iter=new_items.rbegin(); iter < new_items.rend(); ++iter) + for (history_item_list_t::reverse_iterator iter=new_items.rbegin(); iter < new_items.rend(); ++iter) { /* Skip duplicates */ if (! seen.insert(iter->str()).second) @@ -658,7 +698,7 @@ void history_t::get_string_representation(wcstring &result, const wcstring &sepa /* Append old items */ load_old_if_needed(); - for (std::vector::reverse_iterator iter = old_item_offsets.rbegin(); iter != old_item_offsets.rend(); ++iter) + for (std::deque::reverse_iterator iter = old_item_offsets.rbegin(); iter != old_item_offsets.rend(); ++iter) { size_t offset = *iter; const history_item_t item = history_t::decode_item(mmap_start + offset, mmap_length - offset, mmap_type); @@ -825,7 +865,9 @@ history_item_t history_t::decode_item_fish_2_0(const char *base, size_t len) } } done: - return history_item_t(cmd, when, paths); + history_item_t result(cmd, when); + result.required_paths.swap(paths); + return result; } history_item_t history_t::decode_item(const char *base, size_t len, history_file_type_t type) @@ -1280,7 +1322,7 @@ bool history_t::save_internal_via_rewrite() history_lru_cache_t lru(HISTORY_SAVE_MAX); /* Insert old items in, from old to new. Merge them with our new items, inserting items with earlier timestamps first. */ - std::vector::const_iterator new_item_iter = new_items.begin(); + history_item_list_t::const_iterator new_item_iter = new_items.begin(); /* Map in existing items (which may have changed out from underneath us, so don't trust our old mmap'd data) */ const char *local_mmap_start = NULL; @@ -1526,6 +1568,22 @@ void history_t::save_and_vacuum(void) this->save_internal(true); } +void history_t::disable_automatic_saving() +{ + scoped_lock locker(lock); + disable_automatic_save_counter++; + assert(disable_automatic_save_counter != 0); // overflow! +} + +void history_t::enable_automatic_saving() +{ + scoped_lock locker(lock); + assert(disable_automatic_save_counter > 0); //underflow + disable_automatic_save_counter--; + save_internal_unless_disabled(); +} + + void history_t::clear(void) { scoped_lock locker(lock); @@ -1693,11 +1751,10 @@ bool file_detection_context_t::paths_are_valid(const path_list_t &paths) return perform_file_detection(false) > 0; } -file_detection_context_t::file_detection_context_t(history_t *hist, const wcstring &cmd) : +file_detection_context_t::file_detection_context_t(history_t *hist, history_identifier_t ident) : history(hist), - command(cmd), - when(time(NULL)), - working_directory(env_get_pwd_slash()) + working_directory(env_get_pwd_slash()), + history_item_identifier(ident) { } @@ -1710,8 +1767,13 @@ static int threaded_perform_file_detection(file_detection_context_t *ctx) static void perform_file_detection_done(file_detection_context_t *ctx, int success) { - /* Now that file detection is done, create the history item */ - ctx->history->add(ctx->command, ctx->valid_paths); + ASSERT_IS_MAIN_THREAD(); + + /* Now that file detection is done, update the history item with the valid file paths */ + ctx->history->set_valid_file_paths(ctx->valid_paths, ctx->history_item_identifier); + + /* Allow saving again */ + ctx->history->enable_automatic_saving(); /* Done with the context. */ delete ctx; @@ -1721,7 +1783,9 @@ static bool string_could_be_path(const wcstring &potential_path) { // Assume that things with leading dashes aren't paths if (potential_path.empty() || potential_path.at(0) == L'-') + { return false; + } return true; } @@ -1729,44 +1793,73 @@ void history_t::add_with_file_detection(const wcstring &str) { ASSERT_IS_MAIN_THREAD(); path_list_t potential_paths; - - /* Hack hack hack - if the command is likely to trigger an exit, then don't do background file detection, because we won't be able to write it to our history file before we exit. */ + + /* Find all arguments that look like they could be file paths */ bool impending_exit = false; - - tokenizer_t tokenizer(str.c_str(), TOK_SQUASH_ERRORS); - for (; tok_has_next(&tokenizer); tok_next(&tokenizer)) + parse_node_tree_t tree; + parse_tree_from_string(str, parse_flag_none, &tree, NULL); + size_t count = tree.size(); + + for (size_t i=0; i < count; i++) { - int type = tok_last_type(&tokenizer); - if (type == TOK_STRING) + const parse_node_t &node = tree.at(i); + if (! node.has_source()) { - const wchar_t *token_cstr = tok_last(&tokenizer); - if (token_cstr) + continue; + } + + if (node.type == symbol_argument) + { + wcstring potential_path = node.get_source(str); + bool unescaped = unescape_string_in_place(&potential_path, UNESCAPE_DEFAULT); + if (unescaped && string_could_be_path(potential_path)) { - wcstring potential_path; - bool unescaped = unescape_string(token_cstr, &potential_path, UNESCAPE_DEFAULT); - if (unescaped && string_could_be_path(potential_path)) - { - potential_paths.push_back(potential_path); - - /* What a hack! */ - impending_exit = impending_exit || contains(potential_path, L"exec", L"exit", L"reboot"); - } + potential_paths.push_back(potential_path); + } + } + else if (node.type == symbol_plain_statement) + { + /* Hack hack hack - if the command is likely to trigger an exit, then don't do background file detection, because we won't be able to write it to our history file before we exit. */ + if (tree.decoration_for_plain_statement(node) == parse_statement_decoration_exec) + { + impending_exit = true; + } + + wcstring command; + tree.command_for_plain_statement(node, str, &command); + unescape_string_in_place(&command, UNESCAPE_DEFAULT); + if (contains(command, L"exit", L"reboot")) + { + impending_exit = true; } } } - if (potential_paths.empty() || impending_exit) + /* If we got a path, we'll perform file detection for autosuggestion hinting */ + history_identifier_t identifier = 0; + if (! potential_paths.empty() && ! impending_exit) { - this->add(str); - } - else - { - /* We have some paths. Make a context. */ - file_detection_context_t *context = new file_detection_context_t(this, str); - - /* Store the potential paths. Reverse them to put them in the same order as in the command. */ + /* Grab the next identifier */ + static history_identifier_t sLastIdentifier = 0; + identifier = ++sLastIdentifier; + + /* Create a new detection context */ + file_detection_context_t *context = new file_detection_context_t(this, identifier); context->potential_paths.swap(potential_paths); + + /* Prevent saving until we're done, so we have time to get the paths */ + this->disable_automatic_saving(); + + /* Kick it off. Even though we haven't added the item yet, it updates the item on the main thread, so we can't race */ iothread_perform(threaded_perform_file_detection, perform_file_detection_done, context); } -} + /* Actually add the item to the history */ + this->add(str, identifier); + + /* If we think we're about to exit, save immediately, regardless of any disabling. This may cause us to lose file hinting for some commands, but it beats losing history items */ + if (impending_exit) + { + this->save(); + } +} diff --git a/history.h b/history.h index 4b08da678..290d91a89 100644 --- a/history.h +++ b/history.h @@ -9,6 +9,7 @@ #include "common.h" #include "pthread.h" #include "wutil.h" +#include #include #include #include @@ -34,6 +35,8 @@ enum history_search_type_t HISTORY_SEARCH_TYPE_PREFIX }; +typedef uint32_t history_identifier_t; + class history_item_t { friend class history_t; @@ -41,8 +44,8 @@ class history_item_t friend class history_tests_t; private: - explicit history_item_t(const wcstring &); - explicit history_item_t(const wcstring &, time_t, const path_list_t &paths = path_list_t()); + explicit history_item_t(const wcstring &str); + explicit history_item_t(const wcstring &, time_t, history_identifier_t ident = 0); /** Attempts to merge two compatible history items together */ bool merge(const history_item_t &item); @@ -53,9 +56,12 @@ private: /** Original creation time for the entry */ time_t creation_timestamp; + /** Sometimes unique identifier used for hinting */ + history_identifier_t identifier; + /** Paths that we require to be valid for this item to be autosuggested */ path_list_t required_paths; - + public: const wcstring &str() const { @@ -88,6 +94,8 @@ public: } }; +typedef std::deque history_item_list_t; + /* The type of file that we mmap'd */ enum history_file_type_t { @@ -123,10 +131,13 @@ private: const wcstring name; /** New items. Note that these are NOT discarded on save. We need to keep these around so we can distinguish between items in our history and items in the history of other shells that were started after we were started. */ - std::vector new_items; + history_item_list_t new_items; /** The index of the first new item that we have not yet written. */ size_t first_unwritten_new_item_index; + + /** Whether we should disable saving to the file for a time */ + uint32_t disable_automatic_save_counter; /** Deleted item contents. */ std::set deleted_items; @@ -153,7 +164,7 @@ private: void populate_from_mmap(void); /** List of old items, as offsets into out mmap data */ - std::vector old_item_offsets; + std::deque old_item_offsets; /** Whether we've loaded old items */ bool loaded_old; @@ -175,6 +186,9 @@ private: /** Saves history */ void save_internal(bool vacuum); + + /** Saves history, maybe */ + void save_internal_unless_disabled(); /* Do a private, read-only map of the entirety of a history file with the given name. Returns true if successful. Returns the mapped memory region by reference. */ bool map_file(const wcstring &name, const char **out_map_start, size_t *out_map_len, file_id_t *file_id); @@ -195,7 +209,7 @@ public: bool is_empty(void); /** Add a new history item to the end */ - void add(const wcstring &str, const path_list_t &valid_paths = path_list_t()); + void add(const wcstring &str, history_identifier_t ident = 0); /** Remove a history item */ void remove(const wcstring &str); @@ -205,9 +219,13 @@ public: /** Saves history */ void save(); - + /** Performs a full (non-incremental) save */ void save_and_vacuum(); + + /** Enable / disable automatic saving. Main thread only! */ + void disable_automatic_saving(); + void enable_automatic_saving(); /** Irreversibly clears history */ void clear(); @@ -217,6 +235,9 @@ public: /* Gets all the history into a string with ARRAY_SEP_STR. This is intended for the $history environment variable. This may be long! */ void get_string_representation(wcstring &str, const wcstring &separator); + + /** Sets the valid file paths for the history item with the given identifier */ + void set_valid_file_paths(const wcstring_list_t &valid_file_paths, history_identifier_t ident); /** Return the specified history at the specified index. 0 is the index of the current commandline. (So the most recent item is at index 1.) */ history_item_t item_at_index(size_t idx); @@ -295,8 +316,6 @@ public: }; - - /** Init history library. The history file won't actually be loaded until the first time a history search is performed. @@ -316,21 +335,14 @@ void history_sanity_check(); /* A helper class for threaded detection of paths */ struct file_detection_context_t { - /* Constructor */ - file_detection_context_t(history_t *hist, const wcstring &cmd); + file_detection_context_t(history_t *hist, history_identifier_t ident = 0); /* Determine which of potential_paths are valid, and put them in valid_paths */ int perform_file_detection(); /* The history associated with this context */ - history_t *history; - - /* The command */ - wcstring command; - - /* When the command was issued */ - time_t when; + history_t * const history; /* The working directory at the time the command was issued */ wcstring working_directory; @@ -340,6 +352,9 @@ struct file_detection_context_t /* Paths that were found to be valid */ path_list_t valid_paths; + + /* Identifier of the history item to which we are associated */ + const history_identifier_t history_item_identifier; /* Performs file detection. Returns 1 if every path in potential_paths is valid, 0 otherwise. If test_all is true, tests every path; otherwise stops as soon as it reaches an invalid path. */ int perform_file_detection(bool test_all); diff --git a/reader.cpp b/reader.cpp index 3bf970800..69883e16f 100644 --- a/reader.cpp +++ b/reader.cpp @@ -1353,7 +1353,7 @@ struct autosuggestion_context_t search_string(term), cursor_pos(pos), searcher(*history, term, HISTORY_SEARCH_TYPE_PREFIX), - detector(history, term), + detector(history), working_directory(env_get_pwd_slash()), vars(env_vars_snapshot_t::highlighting_keys), generation_count(s_generation_count)