forked from LadybirdBrowser/ladybird
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
10 changed files
with
1,366 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,15 +1,91 @@ | ||
/* | ||
* Copyright (c) 2024, Andrew Kaster <[email protected]> | ||
* Copyright (c) 2024, Altomani Gianluca <[email protected]> | ||
* | ||
* SPDX-License-Identifier: BSD-2-Clause | ||
*/ | ||
|
||
#include <AK/JsonObject.h> | ||
#include <AK/JsonValue.h> | ||
#include <LibJS/Runtime/Array.h> | ||
#include <LibJS/Runtime/Object.h> | ||
#include <LibJS/Runtime/PromiseCapability.h> | ||
#include <LibJS/Runtime/Realm.h> | ||
#include <LibWeb/Crypto/CryptoBindings.h> | ||
#include <LibWeb/WebIDL/DOMException.h> | ||
#include <LibWeb/WebIDL/Types.h> | ||
|
||
namespace Web::Bindings { | ||
#define JWK_PARSE_STRING_PROPERTY(name) \ | ||
if (json_object.has_string(#name##sv)) { \ | ||
key.name = TRY_OR_THROW_OOM(vm, String::from_byte_string(*json_object.get_byte_string(#name##sv))); \ | ||
} | ||
|
||
JS::ThrowCompletionOr<JsonWebKey> JsonWebKey::parse(JS::Realm& realm, ReadonlyBytes data) | ||
{ | ||
auto& vm = realm.vm(); | ||
// 1. Let data be the sequence of bytes to be parsed. | ||
|
||
// 2. Let json be the Unicode string that results from interpreting data according to UTF-8. | ||
// 3. Convert json to UTF-16. | ||
auto json = TRY_OR_THROW_OOM(vm, String::from_utf8(data)); | ||
|
||
// 4. Let result be the object literal that results from executing the JSON.parse internal function | ||
// in the context of a new global object, with text argument set to a JavaScript String containing json. | ||
auto maybe_json_value = JsonValue::from_string(json); | ||
if (maybe_json_value.is_error()) | ||
return vm.throw_completion<WebIDL::SyntaxError>(JS::ErrorType::JsonMalformed); | ||
|
||
auto json_value = maybe_json_value.release_value(); | ||
if (!json_value.is_object()) { | ||
return vm.throw_completion<WebIDL::SyntaxError>("JSON value is not an object"_string); | ||
} | ||
|
||
auto& json_object = json_value.as_object(); | ||
|
||
// 5. Let key be the result of converting result to the IDL dictionary type of JsonWebKey. | ||
JsonWebKey key {}; | ||
JWK_PARSE_STRING_PROPERTY(kty); | ||
JWK_PARSE_STRING_PROPERTY(use); | ||
JWK_PARSE_STRING_PROPERTY(alg); | ||
JWK_PARSE_STRING_PROPERTY(crv); | ||
JWK_PARSE_STRING_PROPERTY(x); | ||
JWK_PARSE_STRING_PROPERTY(y); | ||
JWK_PARSE_STRING_PROPERTY(d); | ||
JWK_PARSE_STRING_PROPERTY(n); | ||
JWK_PARSE_STRING_PROPERTY(e); | ||
JWK_PARSE_STRING_PROPERTY(p); | ||
JWK_PARSE_STRING_PROPERTY(q); | ||
JWK_PARSE_STRING_PROPERTY(dp); | ||
JWK_PARSE_STRING_PROPERTY(dq); | ||
JWK_PARSE_STRING_PROPERTY(qi); | ||
JWK_PARSE_STRING_PROPERTY(k); | ||
|
||
key.ext = json_object.get_bool("ext"sv); | ||
|
||
if (json_object.has_array("key_ops"sv)) { | ||
auto key_ops = *json_object.get_array("key_ops"sv); | ||
key.key_ops = Vector<String> {}; | ||
TRY_OR_THROW_OOM(vm, key.key_ops->try_ensure_capacity(key_ops.size())); | ||
|
||
TRY(key_ops.try_for_each([&key, &vm](auto value) -> JS::Completion { | ||
key.key_ops->append(TRY_OR_THROW_OOM(vm, String::from_byte_string(value.as_string()))); | ||
return {}; | ||
})); | ||
} | ||
|
||
if (json_object.has("oth"sv)) | ||
TODO(); | ||
|
||
// 6. If the kty field of key is not defined, then throw a DataError. | ||
if (!key.kty.has_value()) | ||
return vm.throw_completion<WebIDL::DataError>("kty field is not defined"_string); | ||
|
||
// 7. Return key. | ||
return key; | ||
} | ||
|
||
#undef JWK_PARSE_STRING_PROPERTY | ||
|
||
JS::ThrowCompletionOr<GC::Ref<JS::Object>> JsonWebKey::to_object(JS::Realm& realm) | ||
{ | ||
|
@@ -78,4 +154,70 @@ JS::ThrowCompletionOr<GC::Ref<JS::Object>> JsonWebKey::to_object(JS::Realm& real | |
return object; | ||
} | ||
|
||
ErrorOr<JsonObject> JsonWebKey::to_json() | ||
{ | ||
auto object = JsonObject {}; | ||
|
||
if (kty.has_value()) | ||
object.set("kty", kty->to_byte_string()); | ||
|
||
if (use.has_value()) | ||
object.set("use", use->to_byte_string()); | ||
|
||
if (key_ops.has_value()) { | ||
JsonArray key_ops_array {}; | ||
for (auto& key_usage : key_ops.value()) | ||
TRY(key_ops_array.append(key_usage.to_byte_string())); | ||
object.set("key_ops", move(key_ops_array)); | ||
} | ||
|
||
if (alg.has_value()) | ||
object.set("alg", alg->to_byte_string()); | ||
|
||
if (ext.has_value()) | ||
object.set("ext", *ext); | ||
|
||
if (crv.has_value()) | ||
object.set("crv", crv->to_byte_string()); | ||
|
||
if (x.has_value()) | ||
object.set("x", x->to_byte_string()); | ||
|
||
if (y.has_value()) | ||
object.set("y", y->to_byte_string()); | ||
|
||
if (d.has_value()) | ||
object.set("d", d->to_byte_string()); | ||
|
||
if (n.has_value()) | ||
object.set("n", n->to_byte_string()); | ||
|
||
if (e.has_value()) | ||
object.set("e", e->to_byte_string()); | ||
|
||
if (p.has_value()) | ||
object.set("p", p->to_byte_string()); | ||
|
||
if (q.has_value()) | ||
object.set("q", q->to_byte_string()); | ||
|
||
if (dp.has_value()) | ||
object.set("dp", dp->to_byte_string()); | ||
|
||
if (dq.has_value()) | ||
object.set("dq", dq->to_byte_string()); | ||
|
||
if (qi.has_value()) | ||
object.set("qi", qi->to_byte_string()); | ||
|
||
if (oth.has_value()) { | ||
TODO(); | ||
} | ||
|
||
if (k.has_value()) | ||
object.set("k", k->to_byte_string()); | ||
|
||
return object; | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.