Serenity Operating System
1/*
2 * Copyright (c) 2022-2023, Linus Groh <linusg@serenityos.org>
3 *
4 * SPDX-License-Identifier: BSD-2-Clause
5 */
6
7#include <AK/String.h>
8#include <LibJS/Runtime/AbstractOperations.h>
9#include <LibJS/Runtime/Completion.h>
10#include <LibJS/Runtime/Value.h>
11#include <LibTextCodec/Decoder.h>
12#include <LibWeb/Infra/JSON.h>
13#include <LibWeb/WebIDL/ExceptionOr.h>
14
15namespace Web::Infra {
16
17// https://infra.spec.whatwg.org/#parse-a-json-string-to-a-javascript-value
18WebIDL::ExceptionOr<JS::Value> parse_json_string_to_javascript_value(JS::Realm& realm, StringView string)
19{
20 auto& vm = realm.vm();
21
22 // 1. Return ? Call(%JSON.parse%, undefined, « string »).
23 return TRY(JS::call(vm, realm.intrinsics().json_parse_function(), JS::js_undefined(), MUST_OR_THROW_OOM(JS::PrimitiveString::create(vm, string))));
24}
25
26// https://infra.spec.whatwg.org/#parse-json-bytes-to-a-javascript-value
27WebIDL::ExceptionOr<JS::Value> parse_json_bytes_to_javascript_value(JS::Realm& realm, ReadonlyBytes bytes)
28{
29 auto& vm = realm.vm();
30
31 // 1. Let string be the result of running UTF-8 decode on bytes.
32 TextCodec::UTF8Decoder decoder;
33 auto string = TRY_OR_THROW_OOM(vm, decoder.to_utf8(bytes));
34
35 // 2. Return the result of parsing a JSON string to an Infra value given string.
36 return parse_json_string_to_javascript_value(realm, string);
37}
38
39// https://infra.spec.whatwg.org/#serialize-a-javascript-value-to-a-json-string
40WebIDL::ExceptionOr<String> serialize_javascript_value_to_json_string(JS::VM& vm, JS::Value value)
41{
42 auto& realm = *vm.current_realm();
43
44 // 1. Let result be ? Call(%JSON.stringify%, undefined, « value »).
45 auto result = TRY(JS::call(vm, realm.intrinsics().json_stringify_function(), JS::js_undefined(), value));
46
47 // 2. If result is undefined, then throw a TypeError.
48 if (result.is_undefined())
49 return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Result of stringifying value must not be undefined"sv };
50
51 // 3. Assert: result is a string.
52 VERIFY(result.is_string());
53
54 // 4. Return result.
55 return TRY(result.as_string().utf8_string());
56}
57
58// https://infra.spec.whatwg.org/#serialize-a-javascript-value-to-json-bytes
59WebIDL::ExceptionOr<ByteBuffer> serialize_javascript_value_to_json_bytes(JS::VM& vm, JS::Value value)
60{
61 // 1. Let string be the result of serializing a JavaScript value to a JSON string given value.
62 auto string = TRY(serialize_javascript_value_to_json_string(vm, value));
63
64 // 2. Return the result of running UTF-8 encode on string.
65 // NOTE: LibJS strings are stored as UTF-8.
66 return TRY_OR_THROW_OOM(vm, ByteBuffer::copy(string.bytes()));
67}
68
69}