mirror of
https://github.com/RGBCube/serenity
synced 2025-05-14 10:54:57 +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>
197 lines
7.4 KiB
C++
197 lines
7.4 KiB
C++
/*
|
|
* Copyright (c) 2021, Brandon Scott <xeon.productions@gmail.com>
|
|
* Copyright (c) 2020, Hunter Salyer <thefalsehonesty@gmail.com>
|
|
* Copyright (c) 2021, Sam Atkins <atkinssj@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "WebContentConsoleClient.h"
|
|
#include <LibJS/Interpreter.h>
|
|
#include <LibJS/MarkupGenerator.h>
|
|
#include <LibJS/Script.h>
|
|
#include <LibWeb/Bindings/WindowObject.h>
|
|
#include <WebContent/ConsoleGlobalObject.h>
|
|
|
|
namespace WebContent {
|
|
|
|
WebContentConsoleClient::WebContentConsoleClient(JS::Console& console, WeakPtr<JS::Interpreter> interpreter, ClientConnection& client)
|
|
: ConsoleClient(console)
|
|
, m_client(client)
|
|
, m_interpreter(interpreter)
|
|
{
|
|
JS::DeferGC defer_gc(m_interpreter->heap());
|
|
auto console_global_object = m_interpreter->heap().allocate_without_global_object<ConsoleGlobalObject>(static_cast<Web::Bindings::WindowObject&>(m_interpreter->global_object()));
|
|
console_global_object->initialize_global_object();
|
|
m_console_global_object = JS::make_handle(console_global_object);
|
|
}
|
|
|
|
void WebContentConsoleClient::handle_input(String const& js_source)
|
|
{
|
|
auto script_or_error = JS::Script::parse(js_source, m_interpreter->realm(), "");
|
|
StringBuilder output_html;
|
|
auto result = JS::ThrowCompletionOr<JS::Value> { JS::js_undefined() };
|
|
if (script_or_error.is_error()) {
|
|
auto error = script_or_error.error()[0];
|
|
auto hint = error.source_location_hint(js_source);
|
|
if (!hint.is_empty())
|
|
output_html.append(String::formatted("<pre>{}</pre>", escape_html_entities(hint)));
|
|
result = m_interpreter->vm().throw_completion<JS::SyntaxError>(*m_console_global_object.cell(), error.to_string());
|
|
} else {
|
|
// FIXME: This is not the correct way to do this, we probably want to have
|
|
// multiple execution contexts we switch between.
|
|
auto& global_object_before = m_interpreter->realm().global_object();
|
|
VERIFY(is<Web::Bindings::WindowObject>(global_object_before));
|
|
auto& this_value_before = m_interpreter->realm().global_environment().global_this_value();
|
|
m_interpreter->realm().set_global_object(*m_console_global_object.cell(), &global_object_before);
|
|
|
|
result = m_interpreter->run(script_or_error.value());
|
|
|
|
m_interpreter->realm().set_global_object(global_object_before, &this_value_before);
|
|
}
|
|
|
|
if (result.is_error()) {
|
|
m_interpreter->vm().clear_exception();
|
|
output_html.append("Uncaught exception: ");
|
|
auto error = *result.throw_completion().value();
|
|
if (error.is_object())
|
|
output_html.append(JS::MarkupGenerator::html_from_error(error.as_object()));
|
|
else
|
|
output_html.append(JS::MarkupGenerator::html_from_value(error));
|
|
print_html(output_html.string_view());
|
|
return;
|
|
}
|
|
|
|
print_html(JS::MarkupGenerator::html_from_value(result.value()));
|
|
}
|
|
|
|
void WebContentConsoleClient::print_html(String const& line)
|
|
{
|
|
m_message_log.append({ .type = ConsoleOutput::Type::HTML, .data = line });
|
|
m_client.async_did_output_js_console_message(m_message_log.size() - 1);
|
|
}
|
|
|
|
void WebContentConsoleClient::clear_output()
|
|
{
|
|
m_message_log.append({ .type = ConsoleOutput::Type::Clear, .data = "" });
|
|
m_client.async_did_output_js_console_message(m_message_log.size() - 1);
|
|
}
|
|
|
|
void WebContentConsoleClient::begin_group(String const& label, bool start_expanded)
|
|
{
|
|
m_message_log.append({ .type = start_expanded ? ConsoleOutput::Type::BeginGroup : ConsoleOutput::Type::BeginGroupCollapsed, .data = label });
|
|
m_client.async_did_output_js_console_message(m_message_log.size() - 1);
|
|
}
|
|
|
|
void WebContentConsoleClient::end_group()
|
|
{
|
|
m_message_log.append({ .type = ConsoleOutput::Type::EndGroup, .data = "" });
|
|
m_client.async_did_output_js_console_message(m_message_log.size() - 1);
|
|
}
|
|
|
|
void WebContentConsoleClient::send_messages(i32 start_index)
|
|
{
|
|
// FIXME: Cap the number of messages we send at once?
|
|
auto messages_to_send = m_message_log.size() - start_index;
|
|
if (messages_to_send < 1) {
|
|
// When the console is first created, it requests any messages that happened before
|
|
// then, by requesting with start_index=0. If we don't have any messages at all, that
|
|
// is still a valid request, and we can just ignore it.
|
|
if (start_index != 0)
|
|
m_client.did_misbehave("Requested non-existent console message index.");
|
|
return;
|
|
}
|
|
|
|
// FIXME: Replace with a single Vector of message structs
|
|
Vector<String> message_types;
|
|
Vector<String> messages;
|
|
message_types.ensure_capacity(messages_to_send);
|
|
messages.ensure_capacity(messages_to_send);
|
|
|
|
for (size_t i = start_index; i < m_message_log.size(); i++) {
|
|
auto& message = m_message_log[i];
|
|
switch (message.type) {
|
|
case ConsoleOutput::Type::HTML:
|
|
message_types.append("html");
|
|
break;
|
|
case ConsoleOutput::Type::Clear:
|
|
message_types.append("clear");
|
|
break;
|
|
case ConsoleOutput::Type::BeginGroup:
|
|
message_types.append("group");
|
|
break;
|
|
case ConsoleOutput::Type::BeginGroupCollapsed:
|
|
message_types.append("groupCollapsed");
|
|
break;
|
|
case ConsoleOutput::Type::EndGroup:
|
|
message_types.append("groupEnd");
|
|
break;
|
|
}
|
|
|
|
messages.append(message.data);
|
|
}
|
|
|
|
m_client.async_did_get_js_console_messages(start_index, message_types, messages);
|
|
}
|
|
|
|
void WebContentConsoleClient::clear()
|
|
{
|
|
clear_output();
|
|
}
|
|
|
|
// 2.3. Printer(logLevel, args[, options]), https://console.spec.whatwg.org/#printer
|
|
JS::ThrowCompletionOr<JS::Value> WebContentConsoleClient::printer(JS::Console::LogLevel log_level, PrinterArguments arguments)
|
|
{
|
|
if (log_level == JS::Console::LogLevel::Trace) {
|
|
auto trace = arguments.get<JS::Console::Trace>();
|
|
StringBuilder html;
|
|
if (!trace.label.is_empty())
|
|
html.appendff("<span class='title'>{}</span><br>", escape_html_entities(trace.label));
|
|
|
|
html.append("<span class='trace'>");
|
|
for (auto& function_name : trace.stack)
|
|
html.appendff("-> {}<br>", escape_html_entities(function_name));
|
|
html.append("</span>");
|
|
|
|
print_html(html.string_view());
|
|
return JS::js_undefined();
|
|
}
|
|
|
|
if (log_level == JS::Console::LogLevel::Group || log_level == JS::Console::LogLevel::GroupCollapsed) {
|
|
auto group = arguments.get<JS::Console::Group>();
|
|
begin_group(group.label, log_level == JS::Console::LogLevel::Group);
|
|
return JS::js_undefined();
|
|
}
|
|
|
|
auto output = String::join(" ", arguments.get<Vector<JS::Value>>());
|
|
m_console.output_debug_message(log_level, output);
|
|
|
|
StringBuilder html;
|
|
switch (log_level) {
|
|
case JS::Console::LogLevel::Debug:
|
|
html.append("<span class=\"debug\">(d) ");
|
|
break;
|
|
case JS::Console::LogLevel::Error:
|
|
html.append("<span class=\"error\">(e) ");
|
|
break;
|
|
case JS::Console::LogLevel::Info:
|
|
html.append("<span class=\"info\">(i) ");
|
|
break;
|
|
case JS::Console::LogLevel::Log:
|
|
html.append("<span class=\"log\"> ");
|
|
break;
|
|
case JS::Console::LogLevel::Warn:
|
|
case JS::Console::LogLevel::CountReset:
|
|
html.append("<span class=\"warn\">(w) ");
|
|
break;
|
|
default:
|
|
html.append("<span>");
|
|
break;
|
|
}
|
|
|
|
html.append(escape_html_entities(output));
|
|
html.append("</span>");
|
|
print_html(html.string_view());
|
|
return JS::js_undefined();
|
|
}
|
|
}
|