mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-20 21:32:07 +00:00 
			
		
		
		
	 fd72597999
			
		
	
	
		fd72597999
		
	
	
	
	
		
			
			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;
 | |
| };
 | |
| 
 | |
| }
 |