KeyAlgorithms.cpp 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. /*
  2. * Copyright (c) 2023, stelar7 <dudedbz@gmail.com>
  3. * Copyright (c) 2024, Andrew Kaster <akaster@serenityos.org>
  4. * Copyright (c) 2024, Jelle Raaijmakers <jelle@ladybird.org>
  5. *
  6. * SPDX-License-Identifier: BSD-2-Clause
  7. */
  8. #include <LibJS/Runtime/GlobalObject.h>
  9. #include <LibJS/Runtime/PrimitiveString.h>
  10. #include <LibJS/Runtime/TypedArray.h>
  11. #include <LibWeb/Bindings/ExceptionOrUtils.h>
  12. #include <LibWeb/Crypto/KeyAlgorithms.h>
  13. namespace Web::Crypto {
  14. JS_DEFINE_ALLOCATOR(KeyAlgorithm);
  15. JS_DEFINE_ALLOCATOR(RsaKeyAlgorithm);
  16. JS_DEFINE_ALLOCATOR(RsaHashedKeyAlgorithm);
  17. JS_DEFINE_ALLOCATOR(EcKeyAlgorithm);
  18. JS_DEFINE_ALLOCATOR(AesKeyAlgorithm);
  19. JS_DEFINE_ALLOCATOR(HmacKeyAlgorithm);
  20. template<typename T>
  21. static JS::ThrowCompletionOr<T*> impl_from(JS::VM& vm, StringView Name)
  22. {
  23. auto this_value = vm.this_value();
  24. JS::Object* this_object = nullptr;
  25. if (this_value.is_nullish())
  26. this_object = &vm.current_realm()->global_object();
  27. else
  28. this_object = TRY(this_value.to_object(vm));
  29. if (!is<T>(this_object))
  30. return vm.throw_completion<JS::TypeError>(JS::ErrorType::NotAnObjectOfType, Name);
  31. return static_cast<T*>(this_object);
  32. }
  33. JS::NonnullGCPtr<KeyAlgorithm> KeyAlgorithm::create(JS::Realm& realm)
  34. {
  35. return realm.create<KeyAlgorithm>(realm);
  36. }
  37. KeyAlgorithm::KeyAlgorithm(JS::Realm& realm)
  38. : Object(ConstructWithPrototypeTag::Tag, realm.intrinsics().object_prototype())
  39. , m_realm(realm)
  40. {
  41. }
  42. void KeyAlgorithm::initialize(JS::Realm& realm)
  43. {
  44. define_native_accessor(realm, "name", name_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable);
  45. Base::initialize(realm);
  46. }
  47. JS_DEFINE_NATIVE_FUNCTION(KeyAlgorithm::name_getter)
  48. {
  49. auto* impl = TRY(impl_from<KeyAlgorithm>(vm, "KeyAlgorithm"sv));
  50. auto name = TRY(Bindings::throw_dom_exception_if_needed(vm, [&] { return impl->name(); }));
  51. return JS::PrimitiveString::create(vm, name);
  52. }
  53. void KeyAlgorithm::visit_edges(Visitor& visitor)
  54. {
  55. Base::visit_edges(visitor);
  56. visitor.visit(m_realm);
  57. }
  58. JS::NonnullGCPtr<RsaKeyAlgorithm> RsaKeyAlgorithm::create(JS::Realm& realm)
  59. {
  60. return realm.create<RsaKeyAlgorithm>(realm);
  61. }
  62. RsaKeyAlgorithm::RsaKeyAlgorithm(JS::Realm& realm)
  63. : KeyAlgorithm(realm)
  64. , m_public_exponent(MUST(JS::Uint8Array::create(realm, 0)))
  65. {
  66. }
  67. void RsaKeyAlgorithm::initialize(JS::Realm& realm)
  68. {
  69. Base::initialize(realm);
  70. define_native_accessor(realm, "modulusLength", modulus_length_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable);
  71. define_native_accessor(realm, "publicExponent", public_exponent_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable);
  72. }
  73. void RsaKeyAlgorithm::visit_edges(Visitor& visitor)
  74. {
  75. Base::visit_edges(visitor);
  76. visitor.visit(m_public_exponent);
  77. }
  78. WebIDL::ExceptionOr<void> RsaKeyAlgorithm::set_public_exponent(::Crypto::UnsignedBigInteger exponent)
  79. {
  80. auto& realm = this->realm();
  81. auto& vm = this->vm();
  82. auto bytes = TRY_OR_THROW_OOM(vm, ByteBuffer::create_uninitialized(exponent.trimmed_byte_length()));
  83. bool const remove_leading_zeroes = true;
  84. auto data_size = exponent.export_data(bytes.span(), remove_leading_zeroes);
  85. auto data_slice_be = bytes.bytes().slice(bytes.size() - data_size, data_size);
  86. // The BigInteger typedef from the WebCrypto spec requires the bytes in the Uint8Array be ordered in Big Endian
  87. if constexpr (AK::HostIsLittleEndian) {
  88. Vector<u8, 32> data_slice_le;
  89. data_slice_le.ensure_capacity(data_size);
  90. for (size_t i = 0; i < data_size; ++i) {
  91. data_slice_le.append(data_slice_be[data_size - i - 1]);
  92. }
  93. m_public_exponent = TRY(JS::Uint8Array::create(realm, data_slice_le.size()));
  94. m_public_exponent->viewed_array_buffer()->buffer().overwrite(0, data_slice_le.data(), data_slice_le.size());
  95. } else {
  96. m_public_exponent = TRY(JS::Uint8Array::create(realm, data_slice_be.size()));
  97. m_public_exponent->viewed_array_buffer()->buffer().overwrite(0, data_slice_be.data(), data_slice_be.size());
  98. }
  99. return {};
  100. }
  101. JS_DEFINE_NATIVE_FUNCTION(RsaKeyAlgorithm::modulus_length_getter)
  102. {
  103. auto* impl = TRY(impl_from<RsaKeyAlgorithm>(vm, "RsaKeyAlgorithm"sv));
  104. return JS::Value(impl->modulus_length());
  105. }
  106. JS_DEFINE_NATIVE_FUNCTION(RsaKeyAlgorithm::public_exponent_getter)
  107. {
  108. auto* impl = TRY(impl_from<RsaKeyAlgorithm>(vm, "RsaKeyAlgorithm"sv));
  109. return impl->public_exponent();
  110. }
  111. JS::NonnullGCPtr<EcKeyAlgorithm> EcKeyAlgorithm::create(JS::Realm& realm)
  112. {
  113. return realm.create<EcKeyAlgorithm>(realm);
  114. }
  115. EcKeyAlgorithm::EcKeyAlgorithm(JS::Realm& realm)
  116. : KeyAlgorithm(realm)
  117. {
  118. }
  119. void EcKeyAlgorithm::initialize(JS::Realm& realm)
  120. {
  121. Base::initialize(realm);
  122. define_native_accessor(realm, "namedCurve", named_curve_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable);
  123. }
  124. JS_DEFINE_NATIVE_FUNCTION(EcKeyAlgorithm::named_curve_getter)
  125. {
  126. auto* impl = TRY(impl_from<EcKeyAlgorithm>(vm, "EcKeyAlgorithm"sv));
  127. return JS::PrimitiveString::create(vm, impl->named_curve());
  128. }
  129. JS::NonnullGCPtr<RsaHashedKeyAlgorithm> RsaHashedKeyAlgorithm::create(JS::Realm& realm)
  130. {
  131. return realm.create<RsaHashedKeyAlgorithm>(realm);
  132. }
  133. RsaHashedKeyAlgorithm::RsaHashedKeyAlgorithm(JS::Realm& realm)
  134. : RsaKeyAlgorithm(realm)
  135. , m_hash(String {})
  136. {
  137. }
  138. void RsaHashedKeyAlgorithm::initialize(JS::Realm& realm)
  139. {
  140. Base::initialize(realm);
  141. define_native_accessor(realm, "hash", hash_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable);
  142. }
  143. JS_DEFINE_NATIVE_FUNCTION(RsaHashedKeyAlgorithm::hash_getter)
  144. {
  145. auto* impl = TRY(impl_from<RsaHashedKeyAlgorithm>(vm, "RsaHashedKeyAlgorithm"sv));
  146. auto hash = TRY(Bindings::throw_dom_exception_if_needed(vm, [&] { return impl->hash(); }));
  147. return hash.visit(
  148. [&](String const& hash_string) -> JS::Value {
  149. return JS::PrimitiveString::create(vm, hash_string);
  150. },
  151. [&](JS::Handle<JS::Object> const& hash) -> JS::Value {
  152. return hash;
  153. });
  154. }
  155. JS::NonnullGCPtr<AesKeyAlgorithm> AesKeyAlgorithm::create(JS::Realm& realm)
  156. {
  157. return realm.create<AesKeyAlgorithm>(realm);
  158. }
  159. AesKeyAlgorithm::AesKeyAlgorithm(JS::Realm& realm)
  160. : KeyAlgorithm(realm)
  161. , m_length(0)
  162. {
  163. }
  164. void AesKeyAlgorithm::initialize(JS::Realm& realm)
  165. {
  166. Base::initialize(realm);
  167. define_native_accessor(realm, "length", length_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable);
  168. }
  169. JS_DEFINE_NATIVE_FUNCTION(AesKeyAlgorithm::length_getter)
  170. {
  171. auto* impl = TRY(impl_from<AesKeyAlgorithm>(vm, "AesKeyAlgorithm"sv));
  172. auto length = TRY(Bindings::throw_dom_exception_if_needed(vm, [&] { return impl->length(); }));
  173. return length;
  174. }
  175. JS::NonnullGCPtr<HmacKeyAlgorithm> HmacKeyAlgorithm::create(JS::Realm& realm)
  176. {
  177. return realm.create<HmacKeyAlgorithm>(realm);
  178. }
  179. HmacKeyAlgorithm::HmacKeyAlgorithm(JS::Realm& realm)
  180. : KeyAlgorithm(realm)
  181. {
  182. }
  183. void HmacKeyAlgorithm::initialize(JS::Realm& realm)
  184. {
  185. Base::initialize(realm);
  186. define_native_accessor(realm, "hash", hash_getter, {}, JS::Attribute::Enumerable | JS::Attribute::Configurable);
  187. }
  188. void HmacKeyAlgorithm::visit_edges(JS::Cell::Visitor& visitor)
  189. {
  190. Base::visit_edges(visitor);
  191. visitor.visit(m_hash);
  192. }
  193. JS_DEFINE_NATIVE_FUNCTION(HmacKeyAlgorithm::hash_getter)
  194. {
  195. auto* impl = TRY(impl_from<HmacKeyAlgorithm>(vm, "HmacKeyAlgorithm"sv));
  196. return TRY(Bindings::throw_dom_exception_if_needed(vm, [&] { return impl->hash(); }));
  197. }
  198. JS_DEFINE_NATIVE_FUNCTION(HmacKeyAlgorithm::length_getter)
  199. {
  200. auto* impl = TRY(impl_from<HmacKeyAlgorithm>(vm, "HmacKeyAlgorithm"sv));
  201. return TRY(Bindings::throw_dom_exception_if_needed(vm, [&] { return impl->length(); }));
  202. }
  203. }