mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-05-17 16:42:54 +00:00
This also refactors interpreter creation to follow InitializeHostDefinedRealm, but I couldn't fit it in the title :^) This allows us to follow the spec much more closely rather than being completely ad-hoc with just the parse node instead of having all the surrounding data such as the realm of the parse node. The interpreter creation refactor creates the global execution context once and doesn't take it off the stack. This allows LibWeb to take the global execution context and manually handle it, following the HTML spec. The HTML spec calls this the "realm execution context" of the environment settings object. It also allows us to specify the globalThis type, as it can be different from the global object type. For example, on the web, Window global objects use a WindowProxy global this value to enforce the same origin policy on operations like [[GetOwnProperty]]. Finally, it allows us to directly call Program::execute in perform_eval and perform_shadow_realm_eval as this moves global_declaration_instantiation into Interpreter::run (ScriptEvaluation) as per the spec. Note that this doesn't evalulate Source Text Modules yet or refactor the bytecode interpreter, that's work for future us :^) This patch was originally build by Luke for the environment settings object change but was also needed for modules. So I (davidot) have modified it with the new completion changes and setup for that. Co-authored-by: davidot <davidot@serenityos.org>
238 lines
6.6 KiB
C++
238 lines
6.6 KiB
C++
/*
|
|
* Copyright (c) 2021, Spencer Dixon <spencercdixon@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "Providers.h"
|
|
#include "FuzzyMatch.h"
|
|
#include <AK/LexicalPath.h>
|
|
#include <AK/URL.h>
|
|
#include <LibCore/DirIterator.h>
|
|
#include <LibCore/ElapsedTimer.h>
|
|
#include <LibCore/File.h>
|
|
#include <LibCore/StandardPaths.h>
|
|
#include <LibDesktop/Launcher.h>
|
|
#include <LibGUI/Clipboard.h>
|
|
#include <LibGUI/FileIconProvider.h>
|
|
#include <LibJS/Interpreter.h>
|
|
#include <LibJS/Runtime/GlobalObject.h>
|
|
#include <LibJS/Script.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <serenity.h>
|
|
#include <spawn.h>
|
|
#include <unistd.h>
|
|
|
|
namespace Assistant {
|
|
|
|
void AppResult::activate() const
|
|
{
|
|
if (chdir(Core::StandardPaths::home_directory().characters()) < 0) {
|
|
perror("chdir");
|
|
exit(1);
|
|
}
|
|
|
|
m_app_file->spawn();
|
|
}
|
|
|
|
void CalculatorResult::activate() const
|
|
{
|
|
GUI::Clipboard::the().set_plain_text(title());
|
|
}
|
|
|
|
void FileResult::activate() const
|
|
{
|
|
Desktop::Launcher::open(URL::create_with_file_protocol(title()));
|
|
}
|
|
|
|
void TerminalResult::activate() const
|
|
{
|
|
pid_t pid;
|
|
char const* argv[] = { "Terminal", "-k", "-e", title().characters(), nullptr };
|
|
|
|
if ((errno = posix_spawn(&pid, "/bin/Terminal", nullptr, nullptr, const_cast<char**>(argv), environ))) {
|
|
perror("posix_spawn");
|
|
} else {
|
|
if (disown(pid) < 0)
|
|
perror("disown");
|
|
}
|
|
}
|
|
|
|
void URLResult::activate() const
|
|
{
|
|
Desktop::Launcher::open(URL::create_with_url_or_path(title()));
|
|
}
|
|
|
|
void AppProvider::query(String const& query, Function<void(NonnullRefPtrVector<Result>)> on_complete)
|
|
{
|
|
if (query.starts_with("=") || query.starts_with('$'))
|
|
return;
|
|
|
|
NonnullRefPtrVector<Result> results;
|
|
|
|
Desktop::AppFile::for_each([&](NonnullRefPtr<Desktop::AppFile> app_file) {
|
|
auto match_result = fuzzy_match(query, app_file->name());
|
|
if (!match_result.matched)
|
|
return;
|
|
|
|
auto icon = GUI::FileIconProvider::icon_for_executable(app_file->executable());
|
|
results.append(adopt_ref(*new AppResult(icon.bitmap_for_size(16), app_file->name(), {}, app_file, match_result.score)));
|
|
});
|
|
|
|
on_complete(move(results));
|
|
}
|
|
|
|
void CalculatorProvider::query(String const& query, Function<void(NonnullRefPtrVector<Result>)> on_complete)
|
|
{
|
|
if (!query.starts_with("="))
|
|
return;
|
|
|
|
auto vm = JS::VM::create();
|
|
auto interpreter = JS::Interpreter::create<JS::GlobalObject>(*vm);
|
|
|
|
auto source_code = query.substring(1);
|
|
auto parse_result = JS::Script::parse(source_code, interpreter->realm());
|
|
if (parse_result.is_error())
|
|
return;
|
|
|
|
auto completion = interpreter->run(parse_result.value());
|
|
if (completion.is_error())
|
|
return;
|
|
|
|
auto result = completion.release_value();
|
|
String calculation;
|
|
if (!result.is_number()) {
|
|
calculation = "0";
|
|
} else {
|
|
calculation = result.to_string_without_side_effects();
|
|
}
|
|
|
|
NonnullRefPtrVector<Result> results;
|
|
results.append(adopt_ref(*new CalculatorResult(calculation)));
|
|
on_complete(move(results));
|
|
}
|
|
|
|
Gfx::Bitmap const* FileResult::bitmap() const
|
|
{
|
|
return GUI::FileIconProvider::icon_for_path(title()).bitmap_for_size(16);
|
|
}
|
|
|
|
FileProvider::FileProvider()
|
|
{
|
|
build_filesystem_cache();
|
|
}
|
|
|
|
void FileProvider::query(const String& query, Function<void(NonnullRefPtrVector<Result>)> on_complete)
|
|
{
|
|
build_filesystem_cache();
|
|
|
|
if (m_fuzzy_match_work)
|
|
m_fuzzy_match_work->cancel();
|
|
|
|
m_fuzzy_match_work = Threading::BackgroundAction<NonnullRefPtrVector<Result>>::construct(
|
|
[this, query](auto& task) {
|
|
NonnullRefPtrVector<Result> results;
|
|
|
|
for (auto& path : m_full_path_cache) {
|
|
if (task.is_cancelled())
|
|
return results;
|
|
|
|
auto match_result = fuzzy_match(query, path);
|
|
if (!match_result.matched)
|
|
continue;
|
|
if (match_result.score < 0)
|
|
continue;
|
|
|
|
results.append(adopt_ref(*new FileResult(path, match_result.score)));
|
|
}
|
|
return results;
|
|
},
|
|
[on_complete = move(on_complete)](auto results) {
|
|
on_complete(move(results));
|
|
});
|
|
}
|
|
|
|
void FileProvider::build_filesystem_cache()
|
|
{
|
|
if (m_full_path_cache.size() > 0 || m_building_cache)
|
|
return;
|
|
|
|
m_building_cache = true;
|
|
m_work_queue.enqueue("/");
|
|
|
|
(void)Threading::BackgroundAction<int>::construct(
|
|
[this, strong_ref = NonnullRefPtr(*this)](auto&) {
|
|
String slash = "/";
|
|
auto timer = Core::ElapsedTimer::start_new();
|
|
while (!m_work_queue.is_empty()) {
|
|
auto base_directory = m_work_queue.dequeue();
|
|
|
|
if (base_directory.template is_one_of("/dev"sv, "/proc"sv, "/sys"sv))
|
|
continue;
|
|
|
|
Core::DirIterator di(base_directory, Core::DirIterator::SkipDots);
|
|
|
|
while (di.has_next()) {
|
|
auto path = di.next_path();
|
|
struct stat st = {};
|
|
if (fstatat(di.fd(), path.characters(), &st, AT_SYMLINK_NOFOLLOW) < 0) {
|
|
perror("fstatat");
|
|
continue;
|
|
}
|
|
|
|
if (S_ISLNK(st.st_mode))
|
|
continue;
|
|
|
|
auto full_path = LexicalPath::join(slash, base_directory, path).string();
|
|
|
|
m_full_path_cache.append(full_path);
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
|
m_work_queue.enqueue(full_path);
|
|
}
|
|
}
|
|
}
|
|
dbgln("Built cache in {} ms", timer.elapsed());
|
|
return 0;
|
|
},
|
|
[this](auto) {
|
|
m_building_cache = false;
|
|
});
|
|
}
|
|
|
|
void TerminalProvider::query(String const& query, Function<void(NonnullRefPtrVector<Result>)> on_complete)
|
|
{
|
|
if (!query.starts_with('$'))
|
|
return;
|
|
|
|
auto command = query.substring(1).trim_whitespace();
|
|
|
|
NonnullRefPtrVector<Result> results;
|
|
results.append(adopt_ref(*new TerminalResult(move(command))));
|
|
on_complete(move(results));
|
|
}
|
|
|
|
void URLProvider::query(String const& query, Function<void(NonnullRefPtrVector<Result>)> on_complete)
|
|
{
|
|
if (query.is_empty() || query.starts_with('=') || query.starts_with('$'))
|
|
return;
|
|
|
|
URL url = URL(query);
|
|
|
|
if (url.scheme().is_empty())
|
|
url.set_scheme("http");
|
|
if (url.host().is_empty())
|
|
url.set_host(query);
|
|
if (url.paths().is_empty())
|
|
url.set_paths({ "" });
|
|
|
|
if (!url.is_valid())
|
|
return;
|
|
|
|
NonnullRefPtrVector<Result> results;
|
|
results.append(adopt_ref(*new URLResult(url)));
|
|
on_complete(results);
|
|
}
|
|
|
|
}
|