ladybird/Libraries/LibWeb/WebIDL/ExceptionOr.h
Tim Ledbetter 15fe8e7906 LibWeb: Add formatters for WebIDL exception types
This adds formatters for `WebIDL::Exception`, `WebIDL::SimpleException`
and `WebIDL::DOMException`. These are useful for displaying the content
of errors when debugging.
2025-01-23 21:39:02 +01:00

186 lines
5.6 KiB
C++

/*
* Copyright (c) 2021-2023, Linus Groh <linusg@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#pragma once
#include <AK/NonnullRefPtr.h>
#include <AK/Optional.h>
#include <AK/RefPtr.h>
#include <LibJS/Runtime/Completion.h>
#include <LibWeb/WebIDL/DOMException.h>
namespace Web::WebIDL {
#define ENUMERATE_SIMPLE_WEBIDL_EXCEPTION_TYPES(E) \
E(EvalError) \
E(RangeError) \
E(ReferenceError) \
E(TypeError) \
E(URIError)
#define E(x) x,
enum class SimpleExceptionType {
ENUMERATE_SIMPLE_WEBIDL_EXCEPTION_TYPES(E)
};
#undef E
struct SimpleException {
SimpleExceptionType type;
Variant<String, StringView> message;
};
using Exception = Variant<SimpleException, GC::Ref<DOMException>, JS::Completion>;
template<typename ValueType>
class [[nodiscard]] ExceptionOr {
public:
ExceptionOr()
requires(IsSame<ValueType, Empty>)
: m_result_or_exception(Empty {})
{
}
ExceptionOr(ValueType const& result)
: m_result_or_exception(result)
{
}
ExceptionOr(ValueType&& result)
: m_result_or_exception(move(result))
{
}
// Allows implicit construction of ExceptionOr<T> from a type U if T(U) is a supported constructor.
// Most commonly: Value from Object* or similar, so we can omit the curly braces from "return { TRY(...) };".
// Disabled for POD types to avoid weird conversion shenanigans.
template<typename WrappedValueType>
ExceptionOr(WrappedValueType result)
requires(!IsPOD<ValueType>)
: m_result_or_exception(ValueType { move(result) })
{
}
ExceptionOr(GC::Ref<DOMException> exception)
: m_result_or_exception(exception)
{
}
ExceptionOr(SimpleException exception)
: m_result_or_exception(move(exception))
{
}
ExceptionOr(JS::Completion exception)
: m_result_or_exception(move(exception))
{
auto const& completion = m_result_or_exception.template get<JS::Completion>();
VERIFY(completion.is_error());
}
ExceptionOr(Exception exception)
: m_result_or_exception(move(exception))
{
if (auto* completion = m_result_or_exception.template get_pointer<JS::Completion>())
VERIFY(completion->is_error());
}
ExceptionOr(ExceptionOr&& other) = default;
ExceptionOr(ExceptionOr const& other) = default;
~ExceptionOr() = default;
ValueType& value()
requires(!IsSame<ValueType, Empty>)
{
return m_result_or_exception.template get<ValueType>();
}
ValueType release_value()
{
return move(m_result_or_exception.template get<ValueType>());
}
Exception exception() const
{
return m_result_or_exception.template downcast<SimpleException, GC::Ref<DOMException>, JS::Completion>();
}
bool is_exception() const
{
return !m_result_or_exception.template has<ValueType>();
}
ValueType release_value_but_fixme_should_propagate_errors()
{
VERIFY(!is_error());
return release_value();
}
// These are for compatibility with the TRY() macro in AK.
[[nodiscard]] bool is_error() const { return is_exception(); }
Exception release_error() { return exception(); }
private:
// https://webidl.spec.whatwg.org/#idl-exceptions
Variant<ValueType, SimpleException, GC::Ref<DOMException>, JS::Completion> m_result_or_exception;
};
template<>
class [[nodiscard]] ExceptionOr<void> : public ExceptionOr<Empty> {
public:
using ExceptionOr<Empty>::ExceptionOr;
};
}
namespace AK {
template<>
struct Formatter<Web::WebIDL::SimpleException> : Formatter<StringView> {
ErrorOr<void> format(FormatBuilder& builder, Web::WebIDL::SimpleException const& exception)
{
auto message_view = exception.message.visit(
[](String const& message) -> StringView {
return message.bytes_as_string_view();
},
[](StringView message) -> StringView {
return message;
});
return Formatter<StringView>::format(builder, message_view);
}
};
template<>
struct Formatter<Web::WebIDL::Exception> : Formatter<FormatString> {
ErrorOr<void> format(FormatBuilder& builder, Web::WebIDL::Exception const& exception)
{
return exception.visit(
[&](Web::WebIDL::SimpleException const& simple_exception) -> ErrorOr<void> {
return Formatter<FormatString>::format(builder, "{}"sv, simple_exception);
},
[&](GC::Ref<Web::WebIDL::DOMException> const& dom_exception) -> ErrorOr<void> {
return Formatter<FormatString>::format(builder, "{}"sv, *dom_exception);
},
[&](JS::Completion const& completion) -> ErrorOr<void> {
VERIFY(completion.is_error());
auto value = *completion.value();
if (value.is_object()) {
auto& object = value.as_object();
static const JS::PropertyKey message_property_key { "message" };
auto has_message_or_error = object.has_own_property(message_property_key);
if (!has_message_or_error.is_error() && has_message_or_error.value()) {
auto message_object = object.get_without_side_effects(message_property_key);
return Formatter<StringView>::format(builder, message_object.to_string_without_side_effects());
}
}
return Formatter<StringView>::format(builder, value.to_string_without_side_effects());
});
}
};
}