diff options
author | Tim Ledbetter <timledbetter@gmail.com> | 2023-01-07 17:46:34 +0000 |
---|---|---|
committer | Jelle Raaijmakers <jelle@gmta.nl> | 2023-01-17 21:52:21 +0100 |
commit | d9aa7eacc6b566412598ccf32d432037b4b8a434 (patch) | |
tree | 8ab801d331fef8ebc5e80e5840a0c848e52031b0 /Userland/Applications/Assistant/main.cpp | |
parent | 634d1e0197531e924de71f4f5cb8b7b2357ac578 (diff) | |
download | serenity-d9aa7eacc6b566412598ccf32d432037b4b8a434.zip |
Assistant: Cache results for each query by provider
Previously, results were cached for each query in a single list.
The majority of CPU time was spent determining which items in the
cache had been seen previously. This commit removes the need to
check previous results by holding a separate list of results for each
provider type.
This makes Assistant feel much more responsive to user input,
especially when the filesystem has a lot of files.
Diffstat (limited to 'Userland/Applications/Assistant/main.cpp')
-rw-r--r-- | Userland/Applications/Assistant/main.cpp | 77 |
1 files changed, 45 insertions, 32 deletions
diff --git a/Userland/Applications/Assistant/main.cpp b/Userland/Applications/Assistant/main.cpp index 4be33eaaaa..d2b4ea61ab 100644 --- a/Userland/Applications/Assistant/main.cpp +++ b/Userland/Applications/Assistant/main.cpp @@ -1,11 +1,12 @@ /* * Copyright (c) 2021, Spencer Dixon <spencercdixon@gmail.com> - * Copyright (c) 2022, the SerenityOS developers. + * Copyright (c) 2022-2023, the SerenityOS developers. * * SPDX-License-Identifier: BSD-2-Clause */ #include "Providers.h" +#include <AK/Array.h> #include <AK/DeprecatedString.h> #include <AK/Error.h> #include <AK/LexicalPath.h> @@ -73,46 +74,44 @@ class ResultRow final : public GUI::Button { RefPtr<GUI::Menu> m_context_menu; }; +template<size_t ProviderCount> class Database { public: - explicit Database(AppState& state) + explicit Database(AppState& state, Array<NonnullRefPtr<Provider>, ProviderCount>& providers) : m_state(state) + , m_providers(providers) { - m_providers.append(make_ref_counted<AppProvider>()); - m_providers.append(make_ref_counted<CalculatorProvider>()); - m_providers.append(make_ref_counted<FileProvider>()); - m_providers.append(make_ref_counted<TerminalProvider>()); - m_providers.append(make_ref_counted<URLProvider>()); } Function<void(NonnullRefPtrVector<Result>)> on_new_results; void search(DeprecatedString const& query) { - for (auto& provider : m_providers) { - provider.query(query, [=, this](auto results) { - did_receive_results(query, results); - }); + auto should_display_precached_results = false; + for (size_t i = 0; i < ProviderCount; ++i) { + auto& result_array = m_result_cache.ensure(query); + if (result_array.at(i) == nullptr) { + m_providers[i]->query(query, [this, query, i](auto results) { + { + Threading::MutexLocker db_locker(m_mutex); + auto& result_array = m_result_cache.ensure(query); + if (result_array.at(i) != nullptr) + return; + result_array[i] = make<NonnullRefPtrVector<Result>>(results); + } + on_result_cache_updated(); + }); + } else { + should_display_precached_results = true; + } } + if (should_display_precached_results) + on_result_cache_updated(); } private: - void did_receive_results(DeprecatedString const& query, NonnullRefPtrVector<Result> const& results) + void on_result_cache_updated() { - { - Threading::MutexLocker db_locker(m_mutex); - auto& cache_entry = m_result_cache.ensure(query); - - for (auto& result : results) { - auto found = cache_entry.find_if([&result](auto& other) { - return result.equals(other); - }); - - if (found.is_end()) - cache_entry.append(result); - } - } - Threading::MutexLocker state_locker(m_state.lock); auto new_results = m_result_cache.find(m_state.last_query); if (new_results == m_result_cache.end()) @@ -121,21 +120,28 @@ private: // NonnullRefPtrVector will provide dual_pivot_quick_sort references rather than pointers, // and dual_pivot_quick_sort requires being able to construct the underlying type on the // stack. Assistant::Result is pure virtual, thus cannot be constructed on the stack. - auto& sortable_results = static_cast<Vector<NonnullRefPtr<Result>>&>(new_results->value); + Vector<NonnullRefPtr<Result>> all_results; + for (auto const& results_for_provider : new_results->value) { + if (results_for_provider == nullptr) + continue; + for (auto const& result : *results_for_provider) { + all_results.append(result); + } + } - dual_pivot_quick_sort(sortable_results, 0, static_cast<int>(sortable_results.size() - 1), [](auto& a, auto& b) { + dual_pivot_quick_sort(all_results, 0, static_cast<int>(all_results.size() - 1), [](auto& a, auto& b) { return a->score() > b->score(); }); - on_new_results(new_results->value); + on_new_results(all_results); } AppState& m_state; - NonnullRefPtrVector<Provider> m_providers; + Array<NonnullRefPtr<Provider>, ProviderCount> m_providers; Threading::Mutex m_mutex; - HashMap<DeprecatedString, NonnullRefPtrVector<Result>> m_result_cache; + HashMap<DeprecatedString, Array<OwnPtr<NonnullRefPtrVector<Result>>, ProviderCount>> m_result_cache; }; } @@ -163,7 +169,14 @@ ErrorOr<int> serenity_main(Main::Arguments arguments) window->set_minimizable(false); Assistant::AppState app_state; - Assistant::Database db { app_state }; + Array<NonnullRefPtr<Assistant::Provider>, 5> providers = { + make_ref_counted<Assistant::AppProvider>(), + make_ref_counted<Assistant::CalculatorProvider>(), + make_ref_counted<Assistant::TerminalProvider>(), + make_ref_counted<Assistant::URLProvider>(), + make_ref_counted<Assistant::FileProvider>() + }; + Assistant::Database db { app_state, providers }; auto container = TRY(window->set_main_widget<GUI::Frame>()); container->set_fill_with_background_color(true); |