mirror of
https://github.com/RGBCube/serenity
synced 2025-10-23 23:02:07 +00:00

The WebIDL spec specifies a few "simple" exception types in addition to the DOMException type, let's support all of those. This allows functions returning ExceptionOr<T> to throw regular javascript exceptions (as limited by the webidl spec) by returning a `DOM::SimpleException { DOM::SimpleExceptionType::T, "error message" }` which is pretty damn cool :^)
124 lines
3.1 KiB
C++
124 lines
3.1 KiB
C++
/*
|
|
* Copyright (c) 2021, 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 <LibWeb/DOM/DOMException.h>
|
|
|
|
namespace Web::DOM {
|
|
|
|
#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;
|
|
String message;
|
|
};
|
|
|
|
template<typename ValueType>
|
|
class ExceptionOr {
|
|
public:
|
|
ExceptionOr() requires(IsSame<ValueType, Empty>) = default;
|
|
|
|
ExceptionOr(const ValueType& result)
|
|
: m_result(result)
|
|
{
|
|
}
|
|
|
|
ExceptionOr(ValueType&& result)
|
|
: m_result(move(result))
|
|
{
|
|
}
|
|
|
|
ExceptionOr(NonnullRefPtr<DOMException> exception)
|
|
: m_exception(move(exception))
|
|
{
|
|
}
|
|
|
|
ExceptionOr(SimpleException exception)
|
|
: m_exception(move(exception))
|
|
{
|
|
}
|
|
|
|
ExceptionOr(Variant<SimpleException, NonnullRefPtr<DOMException>> exception)
|
|
: m_exception(move(exception).template downcast<Empty, SimpleException, NonnullRefPtr<DOMException>>())
|
|
{
|
|
}
|
|
|
|
ExceptionOr(ExceptionOr&& other) = default;
|
|
ExceptionOr(const ExceptionOr& other) = default;
|
|
~ExceptionOr() = default;
|
|
|
|
ValueType& value() requires(!IsSame<ValueType, Empty>)
|
|
{
|
|
return m_result.value();
|
|
}
|
|
|
|
ValueType release_value() requires(!IsSame<ValueType, Empty>)
|
|
{
|
|
return m_result.release_value();
|
|
}
|
|
|
|
Variant<SimpleException, NonnullRefPtr<DOMException>> exception() const
|
|
{
|
|
return m_exception.template downcast<SimpleException, NonnullRefPtr<DOMException>>();
|
|
}
|
|
|
|
auto materialized_exception(JS::GlobalObject& global_object) const
|
|
{
|
|
#define E(x) JS::x*,
|
|
using ResultType = Variant<ENUMERATE_SIMPLE_WEBIDL_EXCEPTION_TYPES(E) NonnullRefPtr<DOMException>>;
|
|
#undef E
|
|
|
|
return m_exception.visit(
|
|
[&](SimpleException& exception) -> ResultType {
|
|
switch (exception.type) {
|
|
#define E(x) \
|
|
case SimpleExceptionType::x: \
|
|
return JS::x::create(global_object, exception.message);
|
|
|
|
ENUMERATE_SIMPLE_WEBIDL_EXCEPTION_TYPES(E)
|
|
|
|
#undef E
|
|
default:
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
},
|
|
[&](NonnullRefPtr<DOMException> const& exception) -> ResultType { return exception; },
|
|
[](Empty) -> ResultType { VERIFY_NOT_REACHED(); });
|
|
}
|
|
|
|
bool is_exception() const
|
|
{
|
|
return !m_exception.template has<Empty>();
|
|
}
|
|
|
|
private:
|
|
Optional<ValueType> m_result;
|
|
// https://heycam.github.io/webidl/#idl-exceptions
|
|
Variant<Empty, SimpleException, NonnullRefPtr<DOMException>> m_exception { Empty {} };
|
|
};
|
|
|
|
template<>
|
|
class ExceptionOr<void> : public ExceptionOr<Empty> {
|
|
public:
|
|
using ExceptionOr<Empty>::ExceptionOr;
|
|
};
|
|
|
|
}
|