mirror of
https://github.com/status-im/realm-js.git
synced 2025-01-26 22:39:05 +00:00
23f965060e
* Implement implicit property conversion for date and binary Closes #1542 Closes #1551 * fix include * changelog
391 lines
14 KiB
C++
391 lines
14 KiB
C++
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright 2016 Realm Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#pragma once
|
|
|
|
#include "js_list.hpp"
|
|
#include "js_realm_object.hpp"
|
|
#include "js_schema.hpp"
|
|
|
|
#if REALM_ENABLE_SYNC
|
|
#include <realm/util/base64.hpp>
|
|
#endif
|
|
|
|
namespace realm {
|
|
class List;
|
|
class Object;
|
|
class ObjectSchema;
|
|
class Realm;
|
|
class Results;
|
|
struct Property;
|
|
|
|
namespace js {
|
|
namespace _impl {
|
|
template<typename JSEngine, typename T>
|
|
struct Unbox;
|
|
}
|
|
|
|
template<typename JSEngine>
|
|
class NativeAccessor {
|
|
public:
|
|
using ContextType = typename JSEngine::Context;
|
|
using ObjectType = typename JSEngine::Object;
|
|
using ValueType = typename JSEngine::Value;
|
|
using Object = js::Object<JSEngine>;
|
|
using Value = js::Value<JSEngine>;
|
|
using OptionalValue = util::Optional<ValueType>;
|
|
|
|
NativeAccessor(ContextType ctx, std::shared_ptr<Realm> realm, const ObjectSchema& object_schema)
|
|
: m_ctx(ctx), m_realm(std::move(realm)), m_object_schema(&object_schema) { }
|
|
|
|
template<typename Collection>
|
|
NativeAccessor(ContextType ctx, Collection const& collection)
|
|
: m_ctx(ctx)
|
|
, m_realm(collection.get_realm())
|
|
, m_object_schema(collection.get_type() == realm::PropertyType::Object ? &collection.get_object_schema() : nullptr)
|
|
{ }
|
|
|
|
NativeAccessor(NativeAccessor& parent, const Property& prop)
|
|
: m_ctx(parent.m_ctx)
|
|
, m_realm(parent.m_realm)
|
|
, m_object_schema(&*m_realm->schema().find(prop.object_type))
|
|
{ }
|
|
|
|
OptionalValue value_for_property(ValueType dict, std::string const& prop_name, size_t prop_index) {
|
|
ObjectType object = Value::validated_to_object(m_ctx, dict);
|
|
if (!Object::has_property(m_ctx, object, prop_name)) {
|
|
return util::none;
|
|
}
|
|
ValueType value = Object::get_property(m_ctx, object, prop_name);
|
|
const auto& prop = m_object_schema->persisted_properties[prop_index];
|
|
if (!Value::is_valid_for_property(m_ctx, value, prop)) {
|
|
throw TypeErrorException(*this, m_object_schema->name, prop, value);
|
|
}
|
|
return value;
|
|
}
|
|
|
|
OptionalValue default_value_for_property(const ObjectSchema &object_schema, const std::string &prop_name) {
|
|
auto defaults = get_delegate<JSEngine>(m_realm.get())->m_defaults[object_schema.name];
|
|
auto it = defaults.find(prop_name);
|
|
return it != defaults.end() ? util::make_optional(ValueType(it->second)) : util::none;
|
|
}
|
|
|
|
template<typename T>
|
|
T unbox(ValueType value, bool create = false, bool update = false);
|
|
|
|
template<typename T>
|
|
util::Optional<T> unbox_optional(ValueType value) {
|
|
return is_null(value) ? util::none : util::make_optional(unbox<T>(value));
|
|
}
|
|
|
|
template<typename T>
|
|
ValueType box(util::Optional<T> v) { return v ? box(*v) : null_value(); }
|
|
|
|
ValueType box(bool boolean) { return Value::from_boolean(m_ctx, boolean); }
|
|
ValueType box(int64_t number) { return Value::from_number(m_ctx, number); }
|
|
ValueType box(float number) { return Value::from_number(m_ctx, number); }
|
|
ValueType box(double number) { return Value::from_number(m_ctx, number); }
|
|
ValueType box(StringData string) { return Value::from_string(m_ctx, string.data()); }
|
|
ValueType box(BinaryData data) { return Value::from_binary(m_ctx, data); }
|
|
ValueType box(Mixed) { throw std::runtime_error("'Any' type is unsupported"); }
|
|
|
|
ValueType box(Timestamp ts) {
|
|
if (ts.is_null()) {
|
|
return null_value();
|
|
}
|
|
return Object::create_date(m_ctx, ts.get_seconds() * 1000 + ts.get_nanoseconds() / 1000000);
|
|
}
|
|
ValueType box(realm::Object realm_object) {
|
|
return RealmObjectClass<JSEngine>::create_instance(m_ctx, std::move(realm_object));
|
|
}
|
|
ValueType box(RowExpr row) {
|
|
if (!row.is_attached()) {
|
|
return Value::from_null(m_ctx);
|
|
}
|
|
return RealmObjectClass<JSEngine>::create_instance(m_ctx, realm::Object(m_realm, *m_object_schema, row));
|
|
}
|
|
ValueType box(realm::List list) {
|
|
return ListClass<JSEngine>::create_instance(m_ctx, std::move(list));
|
|
}
|
|
ValueType box(realm::Results results) {
|
|
return ResultsClass<JSEngine>::create_instance(m_ctx, std::move(results));
|
|
}
|
|
|
|
bool is_null(ValueType const& value) {
|
|
return Value::is_null(m_ctx, value) || Value::is_undefined(m_ctx, value);
|
|
}
|
|
ValueType null_value() {
|
|
return Value::from_null(m_ctx);
|
|
}
|
|
|
|
template<typename Fn>
|
|
void enumerate_list(ValueType& value, Fn&& func) {
|
|
auto obj = Value::validated_to_object(m_ctx, value);
|
|
uint32_t size = Object::validated_get_length(m_ctx, obj);
|
|
for (uint32_t i = 0; i < size; ++i) {
|
|
func(Object::get_property(m_ctx, obj, i));
|
|
}
|
|
}
|
|
|
|
bool is_same_list(realm::List const& list, ValueType const& value) const noexcept {
|
|
auto object = Value::validated_to_object(m_ctx, value);
|
|
if (js::Object<JSEngine>::template is_instance<ListClass<JSEngine>>(m_ctx, object)) {
|
|
return list == *get_internal<JSEngine, ListClass<JSEngine>>(object);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool allow_missing(ValueType const&) const noexcept { return false; }
|
|
void will_change(realm::Object&, realm::Property const&) { }
|
|
void did_change() { }
|
|
|
|
std::string print(ValueType const&);
|
|
void print(std::string&, ValueType const&);
|
|
const char *typeof(ValueType const& v) { return Value::typeof(m_ctx, v); }
|
|
|
|
private:
|
|
ContextType m_ctx;
|
|
std::shared_ptr<Realm> m_realm;
|
|
const ObjectSchema* m_object_schema;
|
|
std::string m_string_buffer;
|
|
OwnedBinaryData m_owned_binary_data;
|
|
|
|
template<typename, typename>
|
|
friend struct _impl::Unbox;
|
|
};
|
|
|
|
namespace _impl {
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, bool> {
|
|
static bool call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return js::Value<JSEngine>::validated_to_boolean(ctx->m_ctx, value, "Property");
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, int64_t> {
|
|
static int64_t call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return js::Value<JSEngine>::validated_to_number(ctx->m_ctx, value, "Property");
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, float> {
|
|
static float call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return js::Value<JSEngine>::validated_to_number(ctx->m_ctx, value, "Property");
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, double> {
|
|
static double call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return js::Value<JSEngine>::validated_to_number(ctx->m_ctx, value, "Property");
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, util::Optional<bool>> {
|
|
static util::Optional<bool> call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return ctx->template unbox_optional<bool>(value);
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, util::Optional<int64_t>> {
|
|
static util::Optional<int64_t> call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return ctx->template unbox_optional<int64_t>(value);
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, util::Optional<float>> {
|
|
static util::Optional<float> call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return ctx->template unbox_optional<float>(value);
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, util::Optional<double>> {
|
|
static util::Optional<double> call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
return ctx->template unbox_optional<double>(value);
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, StringData> {
|
|
static StringData call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
if (ctx->is_null(value)) {
|
|
return StringData();
|
|
}
|
|
ctx->m_string_buffer = js::Value<JSEngine>::validated_to_string(ctx->m_ctx, value, "Property");
|
|
return ctx->m_string_buffer;
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, BinaryData> {
|
|
static BinaryData call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value value, bool, bool) {
|
|
if (ctx->is_null(value)) {
|
|
return BinaryData();
|
|
}
|
|
#if REALM_ENABLE_SYNC
|
|
// realm-sync holds the base64-decoding routine
|
|
if (js::Value<JSEngine>::is_string(ctx->m_ctx, value)) {
|
|
// the incoming value might be a base64 string, so let's try to parse it
|
|
std::string str = js::Value<JSEngine>::to_string(ctx->m_ctx, value);
|
|
size_t max_size = util::base64_decoded_size(str.size());
|
|
std::unique_ptr<char[]> data(new char[max_size]);
|
|
if (auto size = util::base64_decode(str, data.get(), max_size)) {
|
|
ctx->m_owned_binary_data = OwnedBinaryData(std::move(data), *size);
|
|
return ctx->m_owned_binary_data.get();
|
|
} else {
|
|
throw std::runtime_error("Attempting to populate BinaryData from string that is not valid base64");
|
|
}
|
|
}
|
|
#endif
|
|
|
|
ctx->m_owned_binary_data = js::Value<JSEngine>::validated_to_binary(ctx->m_ctx, value);
|
|
return ctx->m_owned_binary_data.get();
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, Mixed> {
|
|
static Mixed call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
throw std::runtime_error("'Any' type is unsupported");
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, Timestamp> {
|
|
static Timestamp call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool, bool) {
|
|
if (ctx->is_null(value)) {
|
|
return Timestamp();
|
|
}
|
|
typename JSEngine::Value date;
|
|
if (js::Value<JSEngine>::is_string(ctx->m_ctx, value)) {
|
|
// the incoming value might be a date string, so let the Date constructor have at it
|
|
date = js::Value<JSEngine>::to_date(ctx->m_ctx, value);
|
|
} else {
|
|
date = js::Value<JSEngine>::validated_to_date(ctx->m_ctx, value);
|
|
}
|
|
|
|
double milliseconds = js::Value<JSEngine>::to_number(ctx->m_ctx, date);
|
|
int64_t seconds = milliseconds / 1000;
|
|
int32_t nanoseconds = ((int64_t)milliseconds % 1000) * 1000000;
|
|
return Timestamp(seconds, nanoseconds);
|
|
}
|
|
};
|
|
|
|
template<typename JSEngine>
|
|
struct Unbox<JSEngine, RowExpr> {
|
|
static RowExpr call(NativeAccessor<JSEngine> *ctx, typename JSEngine::Value const& value, bool create, bool try_update) {
|
|
using Value = js::Value<JSEngine>;
|
|
using ValueType = typename JSEngine::Value;
|
|
|
|
auto object = Value::validated_to_object(ctx->m_ctx, value);
|
|
if (js::Object<JSEngine>::template is_instance<RealmObjectClass<JSEngine>>(ctx->m_ctx, object)) {
|
|
auto realm_object = get_internal<JSEngine, RealmObjectClass<JSEngine>>(object);
|
|
if (realm_object->realm() == ctx->m_realm) {
|
|
return realm_object->row();
|
|
}
|
|
if (!create) {
|
|
throw std::runtime_error("Realm object is from another Realm");
|
|
}
|
|
}
|
|
if (!create) {
|
|
throw NonRealmObjectException();
|
|
}
|
|
|
|
if (Value::is_array(ctx->m_ctx, object)) {
|
|
object = Schema<JSEngine>::dict_for_property_array(ctx->m_ctx, *ctx->m_object_schema, object);
|
|
}
|
|
|
|
auto child = realm::Object::create<ValueType>(*ctx, ctx->m_realm, *ctx->m_object_schema,
|
|
static_cast<ValueType>(object), try_update);
|
|
return child.row();
|
|
}
|
|
};
|
|
} // namespace _impl
|
|
|
|
template<typename T>
|
|
template<typename U>
|
|
U NativeAccessor<T>::unbox(ValueType value, bool create, bool update) {
|
|
return _impl::Unbox<T, U>::call(this, std::move(value), create, update);
|
|
}
|
|
|
|
template<typename T>
|
|
std::string NativeAccessor<T>::print(ValueType const& value) {
|
|
std::string ret;
|
|
print(ret, value);
|
|
return ret;
|
|
}
|
|
|
|
template<typename T>
|
|
void NativeAccessor<T>::print(std::string& str, ValueType const& value) {
|
|
if (Value::is_null(m_ctx, value)) {
|
|
str += "null";
|
|
}
|
|
else if (Value::is_undefined(m_ctx, value)) {
|
|
str += "undefined";
|
|
}
|
|
else if (Value::is_array(m_ctx, value)) {
|
|
auto array = Value::to_array(m_ctx, value);
|
|
auto length = Object::validated_get_length(m_ctx, array);
|
|
|
|
str += "[";
|
|
for (uint32_t i = 0; i < length; i++) {
|
|
print(str, Object::get_property(m_ctx, array, i));
|
|
if (i + 1 < length) {
|
|
str += ", ";
|
|
}
|
|
}
|
|
str += "]";
|
|
}
|
|
else if (Value::is_object(m_ctx, value)) {
|
|
auto object = Value::to_object(m_ctx, value);
|
|
if (Object::template is_instance<RealmObjectClass<T>>(m_ctx, object)) {
|
|
auto realm_object = get_internal<T, RealmObjectClass<T>>(object);
|
|
auto& object_schema = realm_object->get_object_schema();
|
|
str += object_schema.name;
|
|
str += "{";
|
|
for (size_t i = 0, count = object_schema.persisted_properties.size(); i < count; ++i) {
|
|
print(str, realm_object->template get_property_value<ValueType>(*this, object_schema.persisted_properties[i].name));
|
|
if (i + 1 < count) {
|
|
str += ", ";
|
|
}
|
|
}
|
|
str += "}";
|
|
}
|
|
else {
|
|
str += Value::to_string(m_ctx, value);
|
|
}
|
|
}
|
|
else if (Value::is_string(m_ctx, value)) {
|
|
str += "'";
|
|
str += Value::to_string(m_ctx, value);
|
|
str += "'";
|
|
}
|
|
else {
|
|
str += Value::to_string(m_ctx, value);
|
|
}
|
|
}
|
|
|
|
} // js
|
|
} // realm
|