mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-04-25 14:05:15 +00:00
This is a clear sign that they want to use a UnixDateTime instead. This also adds support for placing durations and date times into SQL databases via their millisecond offset to UTC.
201 lines
5.6 KiB
C++
201 lines
5.6 KiB
C++
/*
|
|
* Copyright (c) 2021, Jan de Visser <jan@de-visser.net>
|
|
* Copyright (c) 2022, Tim Flynn <trflynn89@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/Checked.h>
|
|
#include <AK/DeprecatedString.h>
|
|
#include <AK/Format.h>
|
|
#include <AK/Optional.h>
|
|
#include <AK/StringView.h>
|
|
#include <AK/Variant.h>
|
|
#include <AK/Vector.h>
|
|
#include <LibIPC/Forward.h>
|
|
#include <LibSQL/Forward.h>
|
|
#include <LibSQL/Result.h>
|
|
#include <LibSQL/Type.h>
|
|
#include <math.h>
|
|
|
|
namespace SQL {
|
|
|
|
template<typename T>
|
|
concept Boolean = SameAs<RemoveCVReference<T>, bool>;
|
|
|
|
template<typename T>
|
|
concept Integer = (Integral<T> && !Boolean<T>);
|
|
|
|
/**
|
|
* A `Value` is an atomic piece of SQL data`. A `Value` has a basic type
|
|
* (Text/String, Integer, Float, etc). Richer types are implemented in higher
|
|
* level layers, but the resulting data is stored in these `Value` objects.
|
|
*/
|
|
class Value {
|
|
template<Integer T>
|
|
using IntegerType = Conditional<IsSigned<T>, i64, u64>;
|
|
|
|
public:
|
|
explicit Value(SQLType sql_type = SQLType::Null);
|
|
explicit Value(DeprecatedString);
|
|
explicit Value(double);
|
|
Value(Value const&);
|
|
Value(Value&&);
|
|
~Value();
|
|
|
|
explicit Value(Integer auto value)
|
|
: m_type(SQLType::Integer)
|
|
, m_value(static_cast<IntegerType<decltype(value)>>(value))
|
|
{
|
|
}
|
|
|
|
explicit Value(Boolean auto value)
|
|
: m_type(SQLType::Boolean)
|
|
, m_value(value)
|
|
{
|
|
}
|
|
|
|
explicit Value(UnixDateTime);
|
|
explicit Value(Duration);
|
|
|
|
static ResultOr<Value> create_tuple(NonnullRefPtr<TupleDescriptor>);
|
|
static ResultOr<Value> create_tuple(Vector<Value>);
|
|
|
|
[[nodiscard]] SQLType type() const;
|
|
[[nodiscard]] StringView type_name() const;
|
|
[[nodiscard]] bool is_type_compatible_with(SQLType) const;
|
|
[[nodiscard]] bool is_null() const;
|
|
[[nodiscard]] bool is_int() const;
|
|
|
|
[[nodiscard]] auto const& value() const
|
|
{
|
|
return *m_value;
|
|
}
|
|
|
|
[[nodiscard]] DeprecatedString to_deprecated_string() const;
|
|
[[nodiscard]] Optional<double> to_double() const;
|
|
[[nodiscard]] Optional<bool> to_bool() const;
|
|
[[nodiscard]] Optional<Vector<Value>> to_vector() const;
|
|
|
|
template<Integer T>
|
|
[[nodiscard]] Optional<T> to_int() const
|
|
{
|
|
if (is_null())
|
|
return {};
|
|
|
|
return m_value->visit(
|
|
[](DeprecatedString const& value) -> Optional<T> {
|
|
if constexpr (IsSigned<T>)
|
|
return value.to_int<T>();
|
|
else
|
|
return value.to_uint<T>();
|
|
},
|
|
[](Integer auto value) -> Optional<T> {
|
|
if (!AK::is_within_range<T>(value))
|
|
return {};
|
|
return static_cast<T>(value);
|
|
},
|
|
[](double value) -> Optional<T> {
|
|
if (!AK::is_within_range<T>(value))
|
|
return {};
|
|
return static_cast<T>(round(value));
|
|
},
|
|
[](bool value) -> Optional<T> { return static_cast<T>(value); },
|
|
[](TupleValue const&) -> Optional<T> { return {}; });
|
|
}
|
|
|
|
Value& operator=(Value);
|
|
Value& operator=(DeprecatedString);
|
|
Value& operator=(double);
|
|
|
|
Value& operator=(Integer auto value)
|
|
{
|
|
m_type = SQLType::Integer;
|
|
m_value = static_cast<IntegerType<decltype(value)>>(value);
|
|
return *this;
|
|
}
|
|
|
|
ResultOr<void> assign_tuple(NonnullRefPtr<TupleDescriptor>);
|
|
ResultOr<void> assign_tuple(Vector<Value>);
|
|
|
|
Value& operator=(Boolean auto value)
|
|
{
|
|
m_type = SQLType::Boolean;
|
|
m_value = value;
|
|
return *this;
|
|
}
|
|
|
|
[[nodiscard]] size_t length() const;
|
|
[[nodiscard]] u32 hash() const;
|
|
void serialize(Serializer&) const;
|
|
void deserialize(Serializer&);
|
|
|
|
[[nodiscard]] int compare(Value const&) const;
|
|
bool operator==(Value const&) const;
|
|
bool operator==(StringView) const;
|
|
bool operator==(double) const;
|
|
|
|
template<Integer T>
|
|
bool operator==(T value)
|
|
{
|
|
return to_int<T>() == value;
|
|
}
|
|
|
|
bool operator!=(Value const&) const;
|
|
bool operator<(Value const&) const;
|
|
bool operator<=(Value const&) const;
|
|
bool operator>(Value const&) const;
|
|
bool operator>=(Value const&) const;
|
|
|
|
ResultOr<Value> add(Value const&) const;
|
|
ResultOr<Value> subtract(Value const&) const;
|
|
ResultOr<Value> multiply(Value const&) const;
|
|
ResultOr<Value> divide(Value const&) const;
|
|
ResultOr<Value> modulo(Value const&) const;
|
|
ResultOr<Value> negate() const;
|
|
ResultOr<Value> shift_left(Value const&) const;
|
|
ResultOr<Value> shift_right(Value const&) const;
|
|
ResultOr<Value> bitwise_or(Value const&) const;
|
|
ResultOr<Value> bitwise_and(Value const&) const;
|
|
ResultOr<Value> bitwise_not() const;
|
|
|
|
[[nodiscard]] TupleElementDescriptor descriptor() const;
|
|
|
|
private:
|
|
friend Serializer;
|
|
|
|
struct TupleValue {
|
|
NonnullRefPtr<TupleDescriptor> descriptor;
|
|
Vector<Value> values;
|
|
};
|
|
|
|
using ValueType = Variant<DeprecatedString, i64, u64, double, bool, TupleValue>;
|
|
|
|
static ResultOr<NonnullRefPtr<TupleDescriptor>> infer_tuple_descriptor(Vector<Value> const& values);
|
|
Value(NonnullRefPtr<TupleDescriptor> descriptor, Vector<Value> values);
|
|
|
|
SQLType m_type { SQLType::Null };
|
|
Optional<ValueType> m_value;
|
|
};
|
|
|
|
}
|
|
|
|
template<>
|
|
struct AK::Formatter<SQL::Value> : Formatter<StringView> {
|
|
ErrorOr<void> format(FormatBuilder& builder, SQL::Value const& value)
|
|
{
|
|
return Formatter<StringView>::format(builder, value.to_deprecated_string());
|
|
}
|
|
};
|
|
|
|
namespace IPC {
|
|
|
|
template<>
|
|
ErrorOr<void> encode(Encoder&, SQL::Value const&);
|
|
|
|
template<>
|
|
ErrorOr<SQL::Value> decode(Decoder&);
|
|
|
|
}
|