2019-06-24 05:25:10 -04:00
|
|
|
#include <AK/Function.h>
|
2019-06-17 13:47:35 -04:00
|
|
|
#include <AK/JsonArray.h>
|
|
|
|
#include <AK/JsonObject.h>
|
2019-06-24 07:38:59 -04:00
|
|
|
#include <AK/JsonParser.h>
|
2019-06-17 13:47:35 -04:00
|
|
|
#include <AK/JsonValue.h>
|
2019-06-17 15:34:12 -04:00
|
|
|
#include <AK/StringBuilder.h>
|
|
|
|
|
|
|
|
namespace AK {
|
2019-06-17 13:47:35 -04:00
|
|
|
|
|
|
|
JsonValue::JsonValue(Type type)
|
|
|
|
: m_type(type)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue::JsonValue(const JsonValue& other)
|
|
|
|
{
|
|
|
|
copy_from(other);
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue& JsonValue::operator=(const JsonValue& other)
|
|
|
|
{
|
|
|
|
if (this != &other) {
|
|
|
|
clear();
|
|
|
|
copy_from(other);
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void JsonValue::copy_from(const JsonValue& other)
|
|
|
|
{
|
|
|
|
m_type = other.m_type;
|
|
|
|
switch (m_type) {
|
|
|
|
case Type::String:
|
|
|
|
m_value.as_string = other.m_value.as_string;
|
2019-06-21 09:29:31 -04:00
|
|
|
AK::ref_if_not_null(m_value.as_string);
|
2019-06-17 13:47:35 -04:00
|
|
|
break;
|
|
|
|
case Type::Object:
|
|
|
|
m_value.as_object = new JsonObject(*other.m_value.as_object);
|
|
|
|
break;
|
|
|
|
case Type::Array:
|
|
|
|
m_value.as_array = new JsonArray(*other.m_value.as_array);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
m_value.as_string = other.m_value.as_string;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue::JsonValue(JsonValue&& other)
|
|
|
|
{
|
|
|
|
m_type = exchange(other.m_type, Type::Undefined);
|
|
|
|
m_value.as_string = exchange(other.m_value.as_string, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue& JsonValue::operator=(JsonValue&& other)
|
|
|
|
{
|
|
|
|
if (this != &other) {
|
2019-06-18 03:22:19 -04:00
|
|
|
clear();
|
2019-06-17 13:47:35 -04:00
|
|
|
m_type = exchange(other.m_type, Type::Undefined);
|
|
|
|
m_value.as_string = exchange(other.m_value.as_string, nullptr);
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue::JsonValue(int value)
|
|
|
|
: m_type(Type::Int)
|
|
|
|
{
|
|
|
|
m_value.as_int = value;
|
|
|
|
}
|
|
|
|
|
2019-06-18 02:55:58 -04:00
|
|
|
JsonValue::JsonValue(unsigned value)
|
|
|
|
{
|
|
|
|
if (value > INT32_MAX) {
|
|
|
|
m_type = Type::Double;
|
|
|
|
m_value.as_double = value;
|
|
|
|
} else {
|
|
|
|
m_type = Type::Int;
|
|
|
|
m_value.as_int = (int)value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-18 03:11:31 -04:00
|
|
|
JsonValue::JsonValue(const char* cstring)
|
|
|
|
: JsonValue(String(cstring))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-06-17 13:47:35 -04:00
|
|
|
JsonValue::JsonValue(double value)
|
|
|
|
: m_type(Type::Double)
|
|
|
|
{
|
|
|
|
m_value.as_double = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue::JsonValue(bool value)
|
|
|
|
: m_type(Type::Bool)
|
|
|
|
{
|
|
|
|
m_value.as_bool = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue::JsonValue(const String& value)
|
|
|
|
{
|
|
|
|
if (value.is_null()) {
|
|
|
|
m_type = Type::Null;
|
|
|
|
} else {
|
|
|
|
m_type = Type::String;
|
|
|
|
m_value.as_string = const_cast<StringImpl*>(value.impl());
|
2019-06-21 09:29:31 -04:00
|
|
|
AK::ref_if_not_null(m_value.as_string);
|
2019-06-17 13:47:35 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue::JsonValue(const JsonObject& value)
|
|
|
|
: m_type(Type::Object)
|
|
|
|
{
|
|
|
|
m_value.as_object = new JsonObject(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
JsonValue::JsonValue(const JsonArray& value)
|
|
|
|
: m_type(Type::Array)
|
|
|
|
{
|
|
|
|
m_value.as_array = new JsonArray(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JsonValue::clear()
|
|
|
|
{
|
|
|
|
switch (m_type) {
|
|
|
|
case Type::String:
|
2019-06-21 09:29:31 -04:00
|
|
|
AK::deref_if_not_null(m_value.as_string);
|
2019-06-17 13:47:35 -04:00
|
|
|
break;
|
|
|
|
case Type::Object:
|
|
|
|
delete m_value.as_object;
|
|
|
|
break;
|
|
|
|
case Type::Array:
|
|
|
|
delete m_value.as_array;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
m_type = Type::Undefined;
|
|
|
|
m_value.as_string = nullptr;
|
|
|
|
}
|
|
|
|
|
2019-06-18 03:37:47 -04:00
|
|
|
void JsonValue::serialize(StringBuilder& builder) const
|
2019-06-17 13:47:35 -04:00
|
|
|
{
|
|
|
|
switch (m_type) {
|
|
|
|
case Type::String:
|
2019-06-17 15:34:12 -04:00
|
|
|
builder.appendf("\"%s\"", m_value.as_string->characters());
|
|
|
|
break;
|
2019-06-17 13:47:35 -04:00
|
|
|
case Type::Array:
|
2019-06-18 03:37:47 -04:00
|
|
|
m_value.as_array->serialize(builder);
|
2019-06-17 15:34:12 -04:00
|
|
|
break;
|
2019-06-17 13:47:35 -04:00
|
|
|
case Type::Object:
|
2019-06-18 03:37:47 -04:00
|
|
|
m_value.as_object->serialize(builder);
|
2019-06-17 15:34:12 -04:00
|
|
|
break;
|
2019-06-17 13:47:35 -04:00
|
|
|
case Type::Bool:
|
2019-06-17 15:34:12 -04:00
|
|
|
builder.append(m_value.as_bool ? "true" : "false");
|
|
|
|
break;
|
2019-06-17 13:47:35 -04:00
|
|
|
case Type::Double:
|
2019-06-17 15:34:12 -04:00
|
|
|
builder.appendf("%g", m_value.as_double);
|
|
|
|
break;
|
2019-06-17 13:47:35 -04:00
|
|
|
case Type::Int:
|
2019-06-17 15:34:12 -04:00
|
|
|
builder.appendf("%d", m_value.as_int);
|
|
|
|
break;
|
2019-06-17 13:47:35 -04:00
|
|
|
case Type::Undefined:
|
2019-06-17 15:34:12 -04:00
|
|
|
builder.append("undefined");
|
|
|
|
break;
|
2019-06-17 13:47:35 -04:00
|
|
|
case Type::Null:
|
2019-06-17 15:34:12 -04:00
|
|
|
builder.append("null");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ASSERT_NOT_REACHED();
|
2019-06-17 13:47:35 -04:00
|
|
|
}
|
2019-06-17 15:34:12 -04:00
|
|
|
}
|
|
|
|
|
2019-06-18 03:37:47 -04:00
|
|
|
String JsonValue::serialized() const
|
2019-06-17 15:34:12 -04:00
|
|
|
{
|
|
|
|
StringBuilder builder;
|
2019-06-18 03:37:47 -04:00
|
|
|
serialize(builder);
|
2019-06-17 15:34:12 -04:00
|
|
|
return builder.to_string();
|
|
|
|
}
|
|
|
|
|
2019-06-24 05:25:10 -04:00
|
|
|
JsonValue JsonValue::from_string(const StringView& input)
|
|
|
|
{
|
2019-06-24 07:38:59 -04:00
|
|
|
return JsonParser(input).parse();
|
2019-06-24 05:25:10 -04:00
|
|
|
}
|
|
|
|
|
2019-06-17 13:47:35 -04:00
|
|
|
}
|