| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2021-04-22 16:53:07 -07:00
										 |  |  |  * Copyright (c) 2020, Matthew Olsson <mattco@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-28 09:17:33 +02:00
										 |  |  | #include <AK/Function.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  | #include <AK/JsonArray.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-20 16:56:01 +02:00
										 |  |  | #include <AK/JsonObject.h>
 | 
					
						
							|  |  |  | #include <AK/JsonParser.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Array.h>
 | 
					
						
							| 
									
										
										
										
											2021-01-01 17:46:39 +01:00
										 |  |  | #include <LibJS/Runtime/BigIntObject.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/BooleanObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | #include <LibJS/Runtime/Error.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/GlobalObject.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/JSONObject.h>
 | 
					
						
							| 
									
										
										
										
											2021-01-01 17:46:39 +01:00
										 |  |  | #include <LibJS/Runtime/NumberObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | #include <LibJS/Runtime/Object.h>
 | 
					
						
							| 
									
										
										
										
											2021-01-01 17:46:39 +01:00
										 |  |  | #include <LibJS/Runtime/StringObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-20 17:11:11 +02:00
										 |  |  | JSONObject::JSONObject(GlobalObject& global_object) | 
					
						
							| 
									
										
										
										
											2020-06-23 17:21:53 +02:00
										 |  |  |     : Object(*global_object.object_prototype()) | 
					
						
							| 
									
										
										
										
											2020-06-20 17:11:11 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-22 17:50:18 +02:00
										 |  |  | void JSONObject::initialize(GlobalObject& global_object) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-10-14 00:03:58 +02:00
										 |  |  |     auto& vm = this->vm(); | 
					
						
							| 
									
										
										
										
											2020-07-22 17:50:18 +02:00
										 |  |  |     Object::initialize(global_object); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     u8 attr = Attribute::Writable | Attribute::Configurable; | 
					
						
							| 
									
										
										
										
											2020-10-14 00:03:58 +02:00
										 |  |  |     define_native_function(vm.names.stringify, stringify, 3, attr); | 
					
						
							|  |  |  |     define_native_function(vm.names.parse, parse, 2, attr); | 
					
						
							|  |  |  |     define_property(vm.well_known_symbol_to_string_tag(), js_string(global_object.heap(), "JSON"), Attribute::Configurable); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | JSONObject::~JSONObject() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | String JSONObject::stringify_impl(GlobalObject& global_object, Value value, Value replacer, Value space) | 
					
						
							| 
									
										
										
										
											2020-08-25 22:18:32 +04:30
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     auto& vm = global_object.vm(); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     StringifyState state; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (replacer.is_object()) { | 
					
						
							|  |  |  |         if (replacer.as_object().is_function()) { | 
					
						
							|  |  |  |             state.replacer_function = &replacer.as_function(); | 
					
						
							|  |  |  |         } else if (replacer.is_array()) { | 
					
						
							|  |  |  |             auto& replacer_object = replacer.as_object(); | 
					
						
							| 
									
										
										
										
											2021-01-10 14:08:27 +01:00
										 |  |  |             auto replacer_length = length_of_array_like(global_object, replacer_object); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |             if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                 return {}; | 
					
						
							|  |  |  |             Vector<String> list; | 
					
						
							|  |  |  |             for (size_t i = 0; i < replacer_length; ++i) { | 
					
						
							|  |  |  |                 auto replacer_value = replacer_object.get(i); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |                 if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                     return {}; | 
					
						
							|  |  |  |                 String item; | 
					
						
							|  |  |  |                 if (replacer_value.is_string() || replacer_value.is_number()) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |                     item = replacer_value.to_string(global_object); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |                     if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                         return {}; | 
					
						
							|  |  |  |                 } else if (replacer_value.is_object()) { | 
					
						
							|  |  |  |                     auto& value_object = replacer_value.as_object(); | 
					
						
							| 
									
										
										
										
											2021-01-01 17:46:39 +01:00
										 |  |  |                     if (is<StringObject>(value_object) || is<NumberObject>(value_object)) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |                         item = value_object.value_of().to_string(global_object); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |                         if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-20 16:56:01 +02:00
										 |  |  |                             return {}; | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (!item.is_null() && !list.contains_slow(item)) { | 
					
						
							|  |  |  |                     list.append(item); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             state.property_list = list; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (space.is_object()) { | 
					
						
							|  |  |  |         auto& space_obj = space.as_object(); | 
					
						
							| 
									
										
										
										
											2021-01-01 17:46:39 +01:00
										 |  |  |         if (is<StringObject>(space_obj) || is<NumberObject>(space_obj)) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             space = space_obj.value_of(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (space.is_number()) { | 
					
						
							|  |  |  |         StringBuilder gap_builder; | 
					
						
							|  |  |  |         auto gap_size = min(10, space.as_i32()); | 
					
						
							|  |  |  |         for (auto i = 0; i < gap_size; ++i) | 
					
						
							|  |  |  |             gap_builder.append(' '); | 
					
						
							|  |  |  |         state.gap = gap_builder.to_string(); | 
					
						
							|  |  |  |     } else if (space.is_string()) { | 
					
						
							|  |  |  |         auto string = space.as_string().string(); | 
					
						
							|  |  |  |         if (string.length() <= 10) { | 
					
						
							|  |  |  |             state.gap = string; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             state.gap = string.substring(0, 10); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         state.gap = String::empty(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-22 17:50:18 +02:00
										 |  |  |     auto* wrapper = Object::create_empty(global_object); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     wrapper->define_property(String::empty(), value); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     auto result = serialize_json_property(global_object, state, String::empty(), wrapper); | 
					
						
							|  |  |  |     if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     if (result.is_null()) | 
					
						
							| 
									
										
										
										
											2020-07-04 11:37:50 -07:00
										 |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | JS_DEFINE_NATIVE_FUNCTION(JSONObject::stringify) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     if (!vm.argument_count()) | 
					
						
							| 
									
										
										
										
											2020-07-04 11:37:50 -07:00
										 |  |  |         return js_undefined(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     auto value = vm.argument(0); | 
					
						
							|  |  |  |     auto replacer = vm.argument(1); | 
					
						
							|  |  |  |     auto space = vm.argument(2); | 
					
						
							| 
									
										
										
										
											2020-07-04 11:37:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     auto string = stringify_impl(global_object, value, replacer, space); | 
					
						
							| 
									
										
										
										
											2020-07-04 11:37:50 -07:00
										 |  |  |     if (string.is_null()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-07-04 11:37:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     return js_string(vm, string); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | String JSONObject::serialize_json_property(GlobalObject& global_object, StringifyState& state, const PropertyName& key, Object* holder) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     auto& vm = global_object.vm(); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     auto value = holder->get(key); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-09-19 12:57:31 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     if (value.is_object()) { | 
					
						
							| 
									
										
										
										
											2020-10-13 23:49:19 +02:00
										 |  |  |         auto to_json = value.as_object().get(vm.names.toJSON); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             return {}; | 
					
						
							|  |  |  |         if (to_json.is_function()) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |             value = vm.call(to_json.as_function(), value, js_string(vm, key.to_string())); | 
					
						
							|  |  |  |             if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                 return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (state.replacer_function) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         value = vm.call(*state.replacer_function, holder, js_string(vm, key.to_string()), value); | 
					
						
							|  |  |  |         if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_object()) { | 
					
						
							|  |  |  |         auto& value_object = value.as_object(); | 
					
						
							| 
									
										
										
										
											2021-01-01 17:46:39 +01:00
										 |  |  |         if (is<NumberObject>(value_object) || is<BooleanObject>(value_object) || is<StringObject>(value_object) || is<BigIntObject>(value_object)) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             value = value_object.value_of(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (value.is_null()) | 
					
						
							|  |  |  |         return "null"; | 
					
						
							|  |  |  |     if (value.is_boolean()) | 
					
						
							|  |  |  |         return value.as_bool() ? "true" : "false"; | 
					
						
							|  |  |  |     if (value.is_string()) | 
					
						
							|  |  |  |         return quote_json_string(value.as_string().string()); | 
					
						
							|  |  |  |     if (value.is_number()) { | 
					
						
							|  |  |  |         if (value.is_finite_number()) | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |             return value.to_string(global_object); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         return "null"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (value.is_object() && !value.is_function()) { | 
					
						
							|  |  |  |         if (value.is_array()) | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |             return serialize_json_array(global_object, state, static_cast<Array&>(value.as_object())); | 
					
						
							|  |  |  |         return serialize_json_object(global_object, state, value.as_object()); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (value.is_bigint()) | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         vm.throw_exception<TypeError>(global_object, ErrorType::JsonBigInt); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | String JSONObject::serialize_json_object(GlobalObject& global_object, StringifyState& state, Object& object) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     auto& vm = global_object.vm(); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     if (state.seen_objects.contains(&object)) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         vm.throw_exception<TypeError>(global_object, ErrorType::JsonCircular); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     state.seen_objects.set(&object); | 
					
						
							|  |  |  |     String previous_indent = state.indent; | 
					
						
							| 
									
										
										
										
											2020-10-04 15:18:52 +01:00
										 |  |  |     state.indent = String::formatted("{}{}", state.indent, state.gap); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     Vector<String> property_strings; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto process_property = [&](const PropertyName& key) { | 
					
						
							| 
									
										
										
										
											2021-04-16 13:02:51 +03:00
										 |  |  |         if (key.is_symbol()) | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         auto serialized_property_string = serialize_json_property(global_object, state, key, &object); | 
					
						
							|  |  |  |         if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             return; | 
					
						
							|  |  |  |         if (!serialized_property_string.is_null()) { | 
					
						
							| 
									
										
										
										
											2020-10-04 15:18:52 +01:00
										 |  |  |             property_strings.append(String::formatted( | 
					
						
							|  |  |  |                 "{}:{}{}", | 
					
						
							|  |  |  |                 quote_json_string(key.to_string()), | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                 state.gap.is_empty() ? "" : " ", | 
					
						
							| 
									
										
										
										
											2020-10-04 15:18:52 +01:00
										 |  |  |                 serialized_property_string)); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (state.property_list.has_value()) { | 
					
						
							|  |  |  |         auto property_list = state.property_list.value(); | 
					
						
							|  |  |  |         for (auto& property : property_list) { | 
					
						
							|  |  |  |             process_property(property); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |             if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                 return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         for (auto& entry : object.indexed_properties()) { | 
					
						
							|  |  |  |             auto value_and_attributes = entry.value_and_attributes(&object); | 
					
						
							|  |  |  |             if (!value_and_attributes.attributes.is_enumerable()) | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             process_property(entry.index()); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |             if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                 return {}; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-20 16:56:01 +02:00
										 |  |  |         for (auto& [key, metadata] : object.shape().property_table_ordered()) { | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             if (!metadata.attributes.is_enumerable()) | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             process_property(key); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |             if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |                 return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  |     if (property_strings.is_empty()) { | 
					
						
							|  |  |  |         builder.append("{}"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         bool first = true; | 
					
						
							|  |  |  |         builder.append('{'); | 
					
						
							|  |  |  |         if (state.gap.is_empty()) { | 
					
						
							|  |  |  |             for (auto& property_string : property_strings) { | 
					
						
							|  |  |  |                 if (!first) | 
					
						
							|  |  |  |                     builder.append(','); | 
					
						
							|  |  |  |                 first = false; | 
					
						
							|  |  |  |                 builder.append(property_string); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             builder.append('\n'); | 
					
						
							|  |  |  |             builder.append(state.indent); | 
					
						
							| 
									
										
										
										
											2020-10-04 15:18:52 +01:00
										 |  |  |             auto separator = String::formatted(",\n{}", state.indent); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             for (auto& property_string : property_strings) { | 
					
						
							|  |  |  |                 if (!first) | 
					
						
							|  |  |  |                     builder.append(separator); | 
					
						
							|  |  |  |                 first = false; | 
					
						
							|  |  |  |                 builder.append(property_string); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             builder.append('\n'); | 
					
						
							|  |  |  |             builder.append(previous_indent); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         builder.append('}'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     state.seen_objects.remove(&object); | 
					
						
							|  |  |  |     state.indent = previous_indent; | 
					
						
							|  |  |  |     return builder.to_string(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | String JSONObject::serialize_json_array(GlobalObject& global_object, StringifyState& state, Object& object) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     auto& vm = global_object.vm(); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     if (state.seen_objects.contains(&object)) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         vm.throw_exception<TypeError>(global_object, ErrorType::JsonCircular); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     state.seen_objects.set(&object); | 
					
						
							|  |  |  |     String previous_indent = state.indent; | 
					
						
							| 
									
										
										
										
											2020-10-04 15:18:52 +01:00
										 |  |  |     state.indent = String::formatted("{}{}", state.indent, state.gap); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |     Vector<String> property_strings; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-10 14:08:27 +01:00
										 |  |  |     auto length = length_of_array_like(global_object, object); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     for (size_t i = 0; i < length; ++i) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         auto serialized_property_string = serialize_json_property(global_object, state, i, &object); | 
					
						
							|  |  |  |         if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             return {}; | 
					
						
							|  |  |  |         if (serialized_property_string.is_null()) { | 
					
						
							|  |  |  |             property_strings.append("null"); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             property_strings.append(serialized_property_string); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  |     if (property_strings.is_empty()) { | 
					
						
							|  |  |  |         builder.append("[]"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (state.gap.is_empty()) { | 
					
						
							|  |  |  |             builder.append('['); | 
					
						
							|  |  |  |             bool first = true; | 
					
						
							|  |  |  |             for (auto& property_string : property_strings) { | 
					
						
							|  |  |  |                 if (!first) | 
					
						
							|  |  |  |                     builder.append(','); | 
					
						
							|  |  |  |                 first = false; | 
					
						
							|  |  |  |                 builder.append(property_string); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             builder.append(']'); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             builder.append("[\n"); | 
					
						
							|  |  |  |             builder.append(state.indent); | 
					
						
							| 
									
										
										
										
											2020-10-04 15:18:52 +01:00
										 |  |  |             auto separator = String::formatted(",\n{}", state.indent); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             bool first = true; | 
					
						
							|  |  |  |             for (auto& property_string : property_strings) { | 
					
						
							|  |  |  |                 if (!first) | 
					
						
							|  |  |  |                     builder.append(separator); | 
					
						
							|  |  |  |                 first = false; | 
					
						
							|  |  |  |                 builder.append(property_string); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             builder.append('\n'); | 
					
						
							|  |  |  |             builder.append(previous_indent); | 
					
						
							|  |  |  |             builder.append(']'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     state.seen_objects.remove(&object); | 
					
						
							|  |  |  |     state.indent = previous_indent; | 
					
						
							|  |  |  |     return builder.to_string(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String JSONObject::quote_json_string(String string) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // FIXME: Handle UTF16
 | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  |     builder.append('"'); | 
					
						
							|  |  |  |     for (auto& ch : string) { | 
					
						
							|  |  |  |         switch (ch) { | 
					
						
							|  |  |  |         case '\b': | 
					
						
							|  |  |  |             builder.append("\\b"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '\t': | 
					
						
							|  |  |  |             builder.append("\\t"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '\n': | 
					
						
							|  |  |  |             builder.append("\\n"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '\f': | 
					
						
							|  |  |  |             builder.append("\\f"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '\r': | 
					
						
							|  |  |  |             builder.append("\\r"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '"': | 
					
						
							|  |  |  |             builder.append("\\\""); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case '\\': | 
					
						
							|  |  |  |             builder.append("\\\\"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             if (ch < 0x20) { | 
					
						
							| 
									
										
										
										
											2021-02-04 00:08:33 +01:00
										 |  |  |                 builder.appendff("\\u{:04x}", ch); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 builder.append(ch); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     builder.append('"'); | 
					
						
							|  |  |  |     return builder.to_string(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-20 13:55:34 +02:00
										 |  |  | JS_DEFINE_NATIVE_FUNCTION(JSONObject::parse) | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     if (!vm.argument_count()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |         return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     auto string = vm.argument(0).to_string(global_object); | 
					
						
							|  |  |  |     if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     auto reviver = vm.argument(1); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto json = JsonValue::from_string(string); | 
					
						
							|  |  |  |     if (!json.has_value()) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |         vm.throw_exception<SyntaxError>(global_object, ErrorType::JsonMalformed); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     Value result = parse_json_value(global_object, json.value()); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     if (reviver.is_function()) { | 
					
						
							| 
									
										
										
										
											2020-07-22 17:50:18 +02:00
										 |  |  |         auto* holder_object = Object::create_empty(global_object); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |         holder_object->define_property(String::empty(), result); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |         if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         return internalize_json_property(global_object, holder_object, String::empty(), reviver.as_function()); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | Value JSONObject::parse_json_value(GlobalObject& global_object, const JsonValue& value) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     if (value.is_object()) | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         return Value(parse_json_object(global_object, value.as_object())); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     if (value.is_array()) | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         return Value(parse_json_array(global_object, value.as_array())); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     if (value.is_null()) | 
					
						
							|  |  |  |         return js_null(); | 
					
						
							|  |  |  |     if (value.is_double()) | 
					
						
							|  |  |  |         return Value(value.as_double()); | 
					
						
							|  |  |  |     if (value.is_number()) | 
					
						
							|  |  |  |         return Value(value.to_i32(0)); | 
					
						
							|  |  |  |     if (value.is_string()) | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         return js_string(global_object.heap(), value.to_string()); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     if (value.is_bool()) | 
					
						
							|  |  |  |         return Value(static_cast<bool>(value.as_bool())); | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |     VERIFY_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | Object* JSONObject::parse_json_object(GlobalObject& global_object, const JsonObject& json_object) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-22 17:50:18 +02:00
										 |  |  |     auto* object = Object::create_empty(global_object); | 
					
						
							| 
									
										
										
										
											2020-06-20 16:56:01 +02:00
										 |  |  |     json_object.for_each_member([&](auto& key, auto& value) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         object->define_property(key, parse_json_value(global_object, value)); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  |     return object; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | Array* JSONObject::parse_json_array(GlobalObject& global_object, const JsonArray& json_array) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-06-20 16:56:01 +02:00
										 |  |  |     auto* array = Array::create(global_object); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     size_t index = 0; | 
					
						
							| 
									
										
										
										
											2020-06-20 16:56:01 +02:00
										 |  |  |     json_array.for_each([&](auto& value) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |         array->define_property(index++, parse_json_value(global_object, value)); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  |     return array; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  | Value JSONObject::internalize_json_property(GlobalObject& global_object, Object* holder, const PropertyName& name, Function& reviver) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     auto& vm = global_object.vm(); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |     auto value = holder->get(name); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     if (value.is_object()) { | 
					
						
							|  |  |  |         auto& value_object = value.as_object(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto process_property = [&](const PropertyName& key) { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |             auto element = internalize_json_property(global_object, &value_object, key, reviver); | 
					
						
							|  |  |  |             if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             if (element.is_undefined()) { | 
					
						
							|  |  |  |                 value_object.delete_property(key); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 value_object.define_property(key, element, default_attributes, false); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (value_object.is_array()) { | 
					
						
							| 
									
										
										
										
											2021-01-10 14:08:27 +01:00
										 |  |  |             auto length = length_of_array_like(global_object, value_object); | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |             for (size_t i = 0; i < length; ++i) { | 
					
						
							|  |  |  |                 process_property(i); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |                 if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |                     return {}; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             for (auto& entry : value_object.indexed_properties()) { | 
					
						
							|  |  |  |                 auto value_and_attributes = entry.value_and_attributes(&value_object); | 
					
						
							|  |  |  |                 if (!value_and_attributes.attributes.is_enumerable()) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 process_property(entry.index()); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |                 if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |                     return {}; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for (auto& [key, metadata] : value_object.shape().property_table_ordered()) { | 
					
						
							|  |  |  |                 if (!metadata.attributes.is_enumerable()) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 process_property(key); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |                 if (vm.exception()) | 
					
						
							| 
									
										
										
										
											2020-06-10 23:30:36 -07:00
										 |  |  |                     return {}; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-08-25 22:18:32 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:52:42 +02:00
										 |  |  |     return vm.call(reviver, Value(holder), js_string(vm, name.to_string()), value); | 
					
						
							| 
									
										
										
										
											2020-06-10 11:01:00 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |