mirror of
https://github.com/RGBCube/serenity
synced 2025-05-14 08:54:58 +00:00
Everywhere: Make JSON serialization fallible
This allows us to eliminate a major source of infallible allocation in the Kernel, as well as lay down the groundwork for OOM fallibility in userland.
This commit is contained in:
parent
6682afb5d4
commit
feb00b7105
18 changed files with 837 additions and 592 deletions
|
@ -1,12 +1,15 @@
|
|||
/*
|
||||
* Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
|
||||
* Copyright (c) 2022, Idan Horowitz <idan.horowitz@serenityos.org>
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-2-Clause
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <AK/Error.h>
|
||||
#include <AK/JsonArraySerializer.h>
|
||||
#include <AK/Try.h>
|
||||
|
||||
#ifndef KERNEL
|
||||
# include <AK/JsonValue.h>
|
||||
|
@ -17,132 +20,217 @@ namespace AK {
|
|||
template<typename Builder>
|
||||
class JsonObjectSerializer {
|
||||
public:
|
||||
explicit JsonObjectSerializer(Builder& builder)
|
||||
: m_builder(builder)
|
||||
static ErrorOr<JsonObjectSerializer> try_create(Builder& builder)
|
||||
{
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(builder.try_append('{'));
|
||||
else
|
||||
TRY(builder.append('{'));
|
||||
return JsonObjectSerializer { builder };
|
||||
}
|
||||
|
||||
JsonObjectSerializer(JsonObjectSerializer&& other)
|
||||
: m_builder(other.m_builder)
|
||||
, m_empty(other.m_empty)
|
||||
, m_finished(exchange(other.m_finished, true))
|
||||
{
|
||||
(void)m_builder.append('{');
|
||||
}
|
||||
|
||||
JsonObjectSerializer(const JsonObjectSerializer&) = delete;
|
||||
JsonObjectSerializer(JsonObjectSerializer&&) = delete;
|
||||
|
||||
~JsonObjectSerializer()
|
||||
{
|
||||
if (!m_finished)
|
||||
finish();
|
||||
VERIFY(m_finished);
|
||||
}
|
||||
|
||||
#ifndef KERNEL
|
||||
void add(StringView key, const JsonValue& value)
|
||||
ErrorOr<void> add(StringView key, const JsonValue& value)
|
||||
{
|
||||
begin_item(key);
|
||||
TRY(begin_item(key));
|
||||
value.serialize(m_builder);
|
||||
return {};
|
||||
}
|
||||
#endif
|
||||
|
||||
void add(StringView key, StringView value)
|
||||
ErrorOr<void> add(StringView key, StringView value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.append('"');
|
||||
(void)m_builder.append_escaped_for_json(value);
|
||||
(void)m_builder.append('"');
|
||||
}
|
||||
|
||||
void add(StringView key, const String& value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.append('"');
|
||||
(void)m_builder.append_escaped_for_json(value);
|
||||
(void)m_builder.append('"');
|
||||
}
|
||||
|
||||
void add(StringView key, const char* value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.append('"');
|
||||
(void)m_builder.append_escaped_for_json(value);
|
||||
(void)m_builder.append('"');
|
||||
}
|
||||
|
||||
void add(StringView key, bool value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.append(value ? "true" : "false");
|
||||
}
|
||||
|
||||
void add(StringView key, int value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.appendff("{}", value);
|
||||
}
|
||||
|
||||
void add(StringView key, unsigned value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.appendff("{}", value);
|
||||
}
|
||||
|
||||
void add(StringView key, long value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.appendff("{}", value);
|
||||
}
|
||||
|
||||
void add(StringView key, long unsigned value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.appendff("{}", value);
|
||||
}
|
||||
|
||||
void add(StringView key, long long value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.appendff("{}", value);
|
||||
}
|
||||
|
||||
void add(StringView key, long long unsigned value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.appendff("{}", value);
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>) {
|
||||
TRY(m_builder.try_append('"'));
|
||||
TRY(m_builder.try_append_escaped_for_json(value));
|
||||
TRY(m_builder.try_append('"'));
|
||||
} else {
|
||||
TRY(m_builder.append('"'));
|
||||
TRY(m_builder.append_escaped_for_json(value));
|
||||
TRY(m_builder.append('"'));
|
||||
}
|
||||
return {};
|
||||
}
|
||||
|
||||
#ifndef KERNEL
|
||||
void add(StringView key, double value)
|
||||
ErrorOr<void> add(StringView key, const String& value)
|
||||
{
|
||||
begin_item(key);
|
||||
(void)m_builder.appendff("{}", value);
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>) {
|
||||
TRY(m_builder.try_append('"'));
|
||||
TRY(m_builder.try_append_escaped_for_json(value));
|
||||
TRY(m_builder.try_append('"'));
|
||||
} else {
|
||||
TRY(m_builder.append('"'));
|
||||
TRY(m_builder.append_escaped_for_json(value));
|
||||
TRY(m_builder.append('"'));
|
||||
}
|
||||
return {};
|
||||
}
|
||||
#endif
|
||||
|
||||
JsonArraySerializer<Builder> add_array(StringView key)
|
||||
ErrorOr<void> add(StringView key, const char* value)
|
||||
{
|
||||
begin_item(key);
|
||||
return JsonArraySerializer(m_builder);
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>) {
|
||||
TRY(m_builder.try_append('"'));
|
||||
TRY(m_builder.try_append_escaped_for_json(value));
|
||||
TRY(m_builder.try_append('"'));
|
||||
} else {
|
||||
TRY(m_builder.append('"'));
|
||||
TRY(m_builder.append_escaped_for_json(value));
|
||||
TRY(m_builder.append('"'));
|
||||
}
|
||||
return {};
|
||||
}
|
||||
|
||||
JsonObjectSerializer<Builder> add_object(StringView key)
|
||||
ErrorOr<void> add(StringView key, bool value)
|
||||
{
|
||||
begin_item(key);
|
||||
return JsonObjectSerializer(m_builder);
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_append(value ? "true" : "false"));
|
||||
else
|
||||
TRY(m_builder.append(value ? "true" : "false"));
|
||||
return {};
|
||||
}
|
||||
|
||||
void finish()
|
||||
ErrorOr<void> add(StringView key, int value)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_appendff("{}", value));
|
||||
else
|
||||
TRY(m_builder.appendff("{}", value));
|
||||
return {};
|
||||
}
|
||||
|
||||
ErrorOr<void> add(StringView key, unsigned value)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_appendff("{}", value));
|
||||
else
|
||||
TRY(m_builder.appendff("{}", value));
|
||||
return {};
|
||||
}
|
||||
|
||||
ErrorOr<void> add(StringView key, long value)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_appendff("{}", value));
|
||||
else
|
||||
TRY(m_builder.appendff("{}", value));
|
||||
return {};
|
||||
}
|
||||
|
||||
ErrorOr<void> add(StringView key, long unsigned value)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_appendff("{}", value));
|
||||
else
|
||||
TRY(m_builder.appendff("{}", value));
|
||||
return {};
|
||||
}
|
||||
|
||||
ErrorOr<void> add(StringView key, long long value)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_appendff("{}", value));
|
||||
else
|
||||
TRY(m_builder.appendff("{}", value));
|
||||
return {};
|
||||
}
|
||||
|
||||
ErrorOr<void> add(StringView key, long long unsigned value)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_appendff("{}", value));
|
||||
else
|
||||
TRY(m_builder.appendff("{}", value));
|
||||
return {};
|
||||
}
|
||||
|
||||
#ifndef KERNEL
|
||||
ErrorOr<void> add(StringView key, double value)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_appendff("{}", value));
|
||||
else
|
||||
TRY(m_builder.appendff("{}", value));
|
||||
return {};
|
||||
}
|
||||
#endif
|
||||
|
||||
ErrorOr<JsonArraySerializer<Builder>> add_array(StringView key)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
return JsonArraySerializer<Builder>::try_create(m_builder);
|
||||
}
|
||||
|
||||
ErrorOr<JsonObjectSerializer<Builder>> add_object(StringView key)
|
||||
{
|
||||
TRY(begin_item(key));
|
||||
return JsonObjectSerializer::try_create(m_builder);
|
||||
}
|
||||
|
||||
ErrorOr<void> finish()
|
||||
{
|
||||
VERIFY(!m_finished);
|
||||
m_finished = true;
|
||||
(void)m_builder.append('}');
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_append('}'));
|
||||
else
|
||||
TRY(m_builder.append('}'));
|
||||
return {};
|
||||
}
|
||||
|
||||
private:
|
||||
void begin_item(StringView key)
|
||||
explicit JsonObjectSerializer(Builder& builder)
|
||||
: m_builder(builder)
|
||||
{
|
||||
if (!m_empty)
|
||||
(void)m_builder.append(',');
|
||||
}
|
||||
|
||||
ErrorOr<void> begin_item(StringView key)
|
||||
{
|
||||
VERIFY(!m_finished);
|
||||
if (!m_empty) {
|
||||
if constexpr (IsLegacyBuilder<Builder>)
|
||||
TRY(m_builder.try_append(','));
|
||||
else
|
||||
TRY(m_builder.append(','));
|
||||
}
|
||||
m_empty = false;
|
||||
|
||||
(void)m_builder.append('"');
|
||||
(void)m_builder.append_escaped_for_json(key);
|
||||
(void)m_builder.append("\":");
|
||||
if constexpr (IsLegacyBuilder<Builder>) {
|
||||
TRY(m_builder.try_append('"'));
|
||||
TRY(m_builder.try_append_escaped_for_json(key));
|
||||
TRY(m_builder.try_append("\":"));
|
||||
} else {
|
||||
TRY(m_builder.append('"'));
|
||||
TRY(m_builder.append_escaped_for_json(key));
|
||||
TRY(m_builder.append("\":"));
|
||||
}
|
||||
return {};
|
||||
}
|
||||
|
||||
Builder& m_builder;
|
||||
|
@ -150,11 +238,21 @@ private:
|
|||
bool m_finished { false };
|
||||
};
|
||||
|
||||
// Template magic to allow for JsonObjectSerializer<>::try_create(...) - Blame CxByte
|
||||
template<>
|
||||
struct JsonObjectSerializer<void> {
|
||||
template<typename Builder>
|
||||
static ErrorOr<JsonObjectSerializer<Builder>> try_create(Builder& builder)
|
||||
{
|
||||
return JsonObjectSerializer<Builder>::try_create(builder);
|
||||
}
|
||||
};
|
||||
|
||||
template<typename Builder>
|
||||
JsonObjectSerializer<Builder> JsonArraySerializer<Builder>::add_object()
|
||||
ErrorOr<JsonObjectSerializer<Builder>> JsonArraySerializer<Builder>::add_object()
|
||||
{
|
||||
begin_item();
|
||||
return JsonObjectSerializer(m_builder);
|
||||
TRY(begin_item());
|
||||
return JsonObjectSerializer<Builder>::try_create(m_builder);
|
||||
}
|
||||
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue