mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-30 21:01:00 +00:00 
			
		
		
		
	
		
			
	
	
		
			153 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			153 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | /*
 | ||
|  |  * Copyright (c) 2023, stelar7 <dudedbz@gmail.com> | ||
|  |  * Copyright (c) 2024, Andrew Kaster <akaster@serenityos.org> | ||
|  |  * | ||
|  |  * SPDX-License-Identifier: BSD-2-Clause | ||
|  |  */ | ||
|  | 
 | ||
|  | #include <LibJS/Runtime/GlobalObject.h>
 | ||
|  | #include <LibJS/Runtime/PrimitiveString.h>
 | ||
|  | #include <LibJS/Runtime/TypedArray.h>
 | ||
|  | #include <LibWeb/Bindings/ExceptionOrUtils.h>
 | ||
|  | #include <LibWeb/Crypto/KeyAlgorithms.h>
 | ||
|  | 
 | ||
|  | namespace Web::Crypto { | ||
|  | 
 | ||
|  | JS_DEFINE_ALLOCATOR(KeyAlgorithm); | ||
|  | JS_DEFINE_ALLOCATOR(RsaKeyAlgorithm); | ||
|  | JS_DEFINE_ALLOCATOR(RsaHashedKeyAlgorithm); | ||
|  | 
 | ||
|  | template<typename T> | ||
|  | static JS::ThrowCompletionOr<T*> impl_from(JS::VM& vm, StringView Name) | ||
|  | { | ||
|  |     auto this_value = vm.this_value(); | ||
|  |     JS::Object* this_object = nullptr; | ||
|  |     if (this_value.is_nullish()) | ||
|  |         this_object = &vm.current_realm()->global_object(); | ||
|  |     else | ||
|  |         this_object = TRY(this_value.to_object(vm)); | ||
|  | 
 | ||
|  |     if (!is<T>(this_object)) | ||
|  |         return vm.throw_completion<JS::TypeError>(JS::ErrorType::NotAnObjectOfType, Name); | ||
|  |     return static_cast<T*>(this_object); | ||
|  | } | ||
|  | 
 | ||
|  | JS::NonnullGCPtr<KeyAlgorithm> KeyAlgorithm::create(JS::Realm& realm) | ||
|  | { | ||
|  |     return realm.heap().allocate<KeyAlgorithm>(realm, realm); | ||
|  | } | ||
|  | 
 | ||
|  | KeyAlgorithm::KeyAlgorithm(JS::Realm& realm) | ||
|  |     : Object(ConstructWithPrototypeTag::Tag, realm.intrinsics().object_prototype()) | ||
|  |     , m_realm(realm) | ||
|  | { | ||
|  | } | ||
|  | 
 | ||
|  | void KeyAlgorithm::initialize(JS::Realm& realm) | ||
|  | { | ||
|  |     define_native_accessor(realm, "name", name_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable); | ||
|  |     Base::initialize(realm); | ||
|  | } | ||
|  | 
 | ||
|  | JS_DEFINE_NATIVE_FUNCTION(KeyAlgorithm::name_getter) | ||
|  | { | ||
|  |     auto* impl = TRY(impl_from<KeyAlgorithm>(vm, "KeyAlgorithm"sv)); | ||
|  |     auto name = TRY(Bindings::throw_dom_exception_if_needed(vm, [&] { return impl->name(); })); | ||
|  |     return JS::PrimitiveString::create(vm, name); | ||
|  | } | ||
|  | 
 | ||
|  | JS::NonnullGCPtr<RsaKeyAlgorithm> RsaKeyAlgorithm::create(JS::Realm& realm) | ||
|  | { | ||
|  |     return realm.heap().allocate<RsaKeyAlgorithm>(realm, realm); | ||
|  | } | ||
|  | 
 | ||
|  | RsaKeyAlgorithm::RsaKeyAlgorithm(JS::Realm& realm) | ||
|  |     : KeyAlgorithm(realm) | ||
|  |     , m_public_exponent(MUST(JS::Uint8Array::create(realm, 0))) | ||
|  | { | ||
|  | } | ||
|  | 
 | ||
|  | void RsaKeyAlgorithm::initialize(JS::Realm& realm) | ||
|  | { | ||
|  |     Base::initialize(realm); | ||
|  | 
 | ||
|  |     define_native_accessor(realm, "modulusLength", modulus_length_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable); | ||
|  |     define_native_accessor(realm, "publicExponent", public_exponent_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable); | ||
|  | } | ||
|  | 
 | ||
|  | void RsaKeyAlgorithm::visit_edges(Visitor& visitor) | ||
|  | { | ||
|  |     Base::visit_edges(visitor); | ||
|  |     visitor.visit(m_public_exponent); | ||
|  | } | ||
|  | 
 | ||
|  | WebIDL::ExceptionOr<void> RsaKeyAlgorithm::set_public_exponent(::Crypto::UnsignedBigInteger exponent) | ||
|  | { | ||
|  |     static_assert(AK::HostIsLittleEndian, "This code assumes a little endian host"); | ||
|  | 
 | ||
|  |     auto& realm = this->realm(); | ||
|  |     auto& vm = this->vm(); | ||
|  | 
 | ||
|  |     auto bytes = TRY_OR_THROW_OOM(vm, ByteBuffer::create_uninitialized(exponent.trimmed_byte_length())); | ||
|  | 
 | ||
|  |     bool const strip_leading_zeroes = true; | ||
|  |     auto data_size = exponent.export_data(bytes.span(), strip_leading_zeroes); | ||
|  | 
 | ||
|  |     // The BigInteger typedef from the WebCrypto spec requires the bytes in the Uint8Array be ordered in Big Endian
 | ||
|  | 
 | ||
|  |     Vector<u8, 32> byte_swapped_data; | ||
|  |     byte_swapped_data.ensure_capacity(data_size); | ||
|  |     for (size_t i = 0; i < data_size; ++i) | ||
|  |         byte_swapped_data.append(bytes[data_size - i - 1]); | ||
|  | 
 | ||
|  |     m_public_exponent = TRY(JS::Uint8Array::create(realm, byte_swapped_data.size())); | ||
|  |     m_public_exponent->viewed_array_buffer()->buffer().overwrite(0, byte_swapped_data.data(), byte_swapped_data.size()); | ||
|  | 
 | ||
|  |     return {}; | ||
|  | } | ||
|  | 
 | ||
|  | JS_DEFINE_NATIVE_FUNCTION(RsaKeyAlgorithm::modulus_length_getter) | ||
|  | { | ||
|  |     auto* impl = TRY(impl_from<RsaKeyAlgorithm>(vm, "RsaKeyAlgorithm"sv)); | ||
|  |     return JS::Value(impl->modulus_length()); | ||
|  | } | ||
|  | 
 | ||
|  | JS_DEFINE_NATIVE_FUNCTION(RsaKeyAlgorithm::public_exponent_getter) | ||
|  | { | ||
|  |     auto* impl = TRY(impl_from<RsaKeyAlgorithm>(vm, "RsaKeyAlgorithm"sv)); | ||
|  |     return impl->public_exponent(); | ||
|  | } | ||
|  | 
 | ||
|  | JS::NonnullGCPtr<RsaHashedKeyAlgorithm> RsaHashedKeyAlgorithm::create(JS::Realm& realm) | ||
|  | { | ||
|  |     return realm.heap().allocate<RsaHashedKeyAlgorithm>(realm, realm); | ||
|  | } | ||
|  | 
 | ||
|  | RsaHashedKeyAlgorithm::RsaHashedKeyAlgorithm(JS::Realm& realm) | ||
|  |     : RsaKeyAlgorithm(realm) | ||
|  |     , m_hash(String {}) | ||
|  | { | ||
|  | } | ||
|  | 
 | ||
|  | void RsaHashedKeyAlgorithm::initialize(JS::Realm& realm) | ||
|  | { | ||
|  |     Base::initialize(realm); | ||
|  | 
 | ||
|  |     define_native_accessor(realm, "hash", hash_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable); | ||
|  | } | ||
|  | 
 | ||
|  | JS_DEFINE_NATIVE_FUNCTION(RsaHashedKeyAlgorithm::hash_getter) | ||
|  | { | ||
|  |     auto* impl = TRY(impl_from<RsaHashedKeyAlgorithm>(vm, "RsaHashedKeyAlgorithm"sv)); | ||
|  |     auto hash = TRY(Bindings::throw_dom_exception_if_needed(vm, [&] { return impl->hash(); })); | ||
|  |     return hash.visit( | ||
|  |         [&](String const& hash_string) -> JS::Value { | ||
|  |             return JS::PrimitiveString::create(vm, hash_string); | ||
|  |         }, | ||
|  |         [&](JS::Handle<JS::Object> const& hash) -> JS::Value { | ||
|  |             return hash; | ||
|  |         }); | ||
|  | } | ||
|  | 
 | ||
|  | } |