realm-js/src/rpc.cpp

466 lines
17 KiB
C++
Raw Normal View History

2016-02-18 11:59:34 -08:00
////////////////////////////////////////////////////////////////////////////
//
// 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.
//
////////////////////////////////////////////////////////////////////////////
2015-10-08 11:57:07 -06:00
2016-04-12 14:42:05 -07:00
#include <cassert>
#include <dlfcn.h>
2015-10-08 11:57:07 -06:00
#include <map>
#include <string>
2016-04-12 14:42:05 -07:00
#include "rpc.hpp"
#include "jsc_init.hpp"
2015-10-08 11:57:07 -06:00
#include "base64.hpp"
2015-10-08 11:57:07 -06:00
#include "object_accessor.hpp"
#include "shared_realm.hpp"
#include "results.hpp"
2016-04-12 14:42:05 -07:00
using namespace realm;
using namespace realm::rpc;
using Accessor = NativeAccessor<JSValueRef, JSContextRef>;
2015-10-08 11:57:07 -06:00
static const char * const RealmObjectTypesData = "data";
static const char * const RealmObjectTypesDate = "date";
static const char * const RealmObjectTypesDictionary = "dict";
static const char * const RealmObjectTypesFunction = "function";
static const char * const RealmObjectTypesList = "list";
static const char * const RealmObjectTypesObject = "object";
static const char * const RealmObjectTypesResults = "results";
static const char * const RealmObjectTypesRealm = "realm";
static const char * const RealmObjectTypesUndefined = "undefined";
static RPCServer*& get_rpc_server(JSGlobalContextRef ctx) {
static std::map<JSGlobalContextRef, RPCServer*> s_map;
return s_map[ctx];
}
RPCWorker::RPCWorker() {
m_thread = std::thread([this]() {
// TODO: Create ALooper/CFRunLoop to support async calls.
while (!m_stop) {
try_run_task();
}
});
}
RPCWorker::~RPCWorker() {
m_stop = true;
m_thread.join();
}
void RPCWorker::add_task(std::function<json()> task) {
m_tasks.push_back(std::packaged_task<json()>(task));
}
json RPCWorker::pop_task_result() {
// This might block until a future has been added.
auto future = m_futures.pop_back();
// This will block until a return value (or exception) is available.
return future.get();
}
void RPCWorker::try_run_task() {
try {
// Use a 10 millisecond timeout to keep this thread unblocked.
auto task = m_tasks.pop_back(10);
task();
// Since this can be called recursively, it must be pushed to the front of the queue *after* running the task.
m_futures.push_front(task.get_future());
}
catch (ConcurrentDequeTimeout &) {
// We tried.
}
}
2015-10-22 10:44:10 -07:00
RPCServer::RPCServer() {
2015-10-22 17:59:05 -07:00
m_context = JSGlobalContextCreate(NULL);
get_rpc_server(m_context) = this;
2015-10-08 11:57:07 -06:00
2015-10-22 10:44:10 -07:00
// JavaScriptCore crashes when trying to walk up the native stack to print the stacktrace.
// FIXME: Avoid having to do this!
static void (*setIncludesNativeCallStack)(JSGlobalContextRef, bool) = (void (*)(JSGlobalContextRef, bool))dlsym(RTLD_DEFAULT, "JSGlobalContextSetIncludesNativeCallStackWhenReportingExceptions");
if (setIncludesNativeCallStack) {
2015-10-22 17:59:05 -07:00
setIncludesNativeCallStack(m_context, false);
2015-10-15 18:50:20 -07:00
}
2015-10-22 17:59:05 -07:00
m_requests["/create_session"] = [this](const json dict) {
RJSInitializeInContext(m_context);
2015-10-15 18:50:20 -07:00
2016-04-12 14:42:05 -07:00
jsc::String realm_string = "Realm";
JSObjectRef realm_constructor = jsc::Object::validated_get_constructor(m_context, JSContextGetGlobalObject(m_context), realm_string);
2015-10-22 17:59:05 -07:00
m_session_id = store_object(realm_constructor);
return (json){{"result", m_session_id}};
2015-10-22 10:44:10 -07:00
};
2015-10-22 17:59:05 -07:00
m_requests["/create_realm"] = [this](const json dict) {
JSObjectRef realm_constructor = m_session_id ? JSObjectRef(m_objects[m_session_id]) : NULL;
2015-10-22 10:44:10 -07:00
if (!realm_constructor) {
throw std::runtime_error("Realm constructor not found!");
}
2015-10-22 10:44:10 -07:00
json::array_t args = dict["arguments"];
size_t arg_count = args.size();
JSValueRef arg_values[arg_count];
2015-10-15 18:50:20 -07:00
2015-10-22 10:44:10 -07:00
for (size_t i = 0; i < arg_count; i++) {
arg_values[i] = deserialize_json_value(args[i]);
}
2016-04-12 14:42:05 -07:00
JSObjectRef realm_object = jsc::Function::construct(m_context, realm_constructor, arg_count, arg_values);
2015-10-22 17:59:05 -07:00
RPCObjectID realm_id = store_object(realm_object);
return (json){{"result", realm_id}};
2015-10-22 10:44:10 -07:00
};
2015-10-22 17:59:05 -07:00
m_requests["/call_method"] = [this](const json dict) {
JSObjectRef object = m_objects[dict["id"].get<RPCObjectID>()];
2016-04-12 14:42:05 -07:00
std::string method_string = dict["name"].get<std::string>();
JSObjectRef function = jsc::Object::validated_get_function(m_context, object, method_string);
2015-10-22 10:44:10 -07:00
json args = dict["arguments"];
2016-04-12 14:42:05 -07:00
size_t arg_count = args.size();
JSValueRef arg_values[arg_count];
for (size_t i = 0; i < arg_count; i++) {
2015-10-22 17:59:05 -07:00
arg_values[i] = deserialize_json_value(args[i]);
2015-10-22 10:44:10 -07:00
}
2016-04-12 14:42:05 -07:00
JSValueRef result = jsc::Function::call(m_context, function, object, arg_count, arg_values);
2015-10-22 16:49:32 -07:00
return (json){{"result", serialize_json_value(result)}};
2015-10-22 10:44:10 -07:00
};
2015-10-22 17:59:05 -07:00
m_requests["/get_property"] = [this](const json dict) {
2015-10-22 10:44:10 -07:00
RPCObjectID oid = dict["id"].get<RPCObjectID>();
json name = dict["name"];
2016-04-12 14:42:05 -07:00
JSValueRef value;
2015-10-22 10:44:10 -07:00
if (name.is_number()) {
2016-04-12 14:42:05 -07:00
value = jsc::Object::get_property(m_context, m_objects[oid], name.get<unsigned int>());
2015-10-22 10:44:10 -07:00
}
else {
2016-04-12 14:42:05 -07:00
value = jsc::Object::get_property(m_context, m_objects[oid], name.get<std::string>());
2015-10-22 10:44:10 -07:00
}
2015-10-22 16:49:32 -07:00
return (json){{"result", serialize_json_value(value)}};
2015-10-22 10:44:10 -07:00
};
2015-10-22 17:59:05 -07:00
m_requests["/set_property"] = [this](const json dict) {
2015-10-22 10:44:10 -07:00
RPCObjectID oid = dict["id"].get<RPCObjectID>();
json name = dict["name"];
JSValueRef value = deserialize_json_value(dict["value"]);
if (name.is_number()) {
2016-04-12 14:42:05 -07:00
jsc::Object::set_property(m_context, m_objects[oid], name.get<unsigned int>(), value);
2015-10-22 10:44:10 -07:00
}
else {
2016-04-12 14:42:05 -07:00
jsc::Object::set_property(m_context, m_objects[oid], name.get<std::string>(), value);
2015-10-22 10:44:10 -07:00
}
2015-10-22 16:49:32 -07:00
return json::object();
2015-10-22 10:44:10 -07:00
};
2015-10-22 17:59:05 -07:00
m_requests["/dispose_object"] = [this](const json dict) {
2015-10-22 10:44:10 -07:00
RPCObjectID oid = dict["id"].get<RPCObjectID>();
2015-10-22 17:59:05 -07:00
m_objects.erase(oid);
2015-10-22 16:49:32 -07:00
return json::object();
2015-10-22 10:44:10 -07:00
};
2015-10-22 17:59:05 -07:00
m_requests["/clear_test_state"] = [this](const json dict) {
for (auto object : m_objects) {
2015-10-22 10:44:10 -07:00
// The session ID points to the Realm constructor object, which should remain.
if (object.first != m_session_id) {
m_objects.erase(object.first);
2015-10-08 11:57:07 -06:00
}
2015-10-22 10:44:10 -07:00
}
m_callbacks.clear();
2015-10-22 17:59:05 -07:00
JSGarbageCollect(m_context);
js::delete_all_realms();
2015-10-22 16:49:32 -07:00
return json::object();
2015-10-22 10:44:10 -07:00
};
}
2015-10-15 17:07:10 -07:00
2015-10-22 10:44:10 -07:00
RPCServer::~RPCServer() {
// The protected values should be unprotected before releasing the context.
m_objects.clear();
m_callbacks.clear();
get_rpc_server(m_context) = nullptr;
2015-10-22 17:59:05 -07:00
JSGlobalContextRelease(m_context);
2015-10-08 11:57:07 -06:00
}
void RPCServer::run_callback(JSContextRef ctx, JSObjectRef this_object, size_t argc, const JSValueRef arguments[], jsc::ReturnValue &return_value) {
RPCServer* server = get_rpc_server(JSContextGetGlobalContext(ctx));
if (!server) {
return;
}
// The first argument was curried to be the callback id.
RPCObjectID callback_id = jsc::Value::to_number(ctx, arguments[0]);
JSObjectRef arguments_array = jsc::Object::create_array(ctx, uint32_t(argc - 1), argc == 1 ? nullptr : arguments + 1);
json arguments_json = server->serialize_json_value(arguments_array);
// The next task on the stack will instruct the JS to run this callback.
// This captures references since it will be executed before exiting this function.
server->m_worker.add_task([&]() -> json {
return {
{"callback", callback_id},
{"arguments", arguments_json},
};
});
// Wait for the next callback result to come off the result stack.
while (server->m_callback_results.empty()) {
// This may recursively bring us into another callback, hence the callback results being a stack.
server->m_worker.try_run_task();
}
json results = server->m_callback_results.pop_back();
json error = results["error"];
// The callback id should be identical!
assert(callback_id == results["callback"].get<RPCObjectID>());
if (!error.is_null()) {
throw jsc::Exception(ctx, error.get<std::string>());
}
return_value.set(server->deserialize_json_value(results["result"]));
}
json RPCServer::perform_request(std::string name, const json &args) {
std::lock_guard<std::mutex> lock(m_request_mutex);
2015-10-22 18:06:11 -07:00
// Only create_session is allowed without the correct session id (since it creates the session id).
if (name != "/create_session" && m_session_id != args["sessionId"].get<RPCObjectID>()) {
return {{"error", "Invalid session ID"}};
}
// The callback_result message contains the return value (or exception) of a callback ran by run_callback().
if (name == "/callback_result") {
json results(args);
m_callback_results.push_back(std::move(results));
}
else {
RPCRequest action = m_requests[name];
assert(action);
m_worker.add_task([=] {
return action(args);
});
}
try {
// This will either be the return value (or exception) of the action perform, OR an instruction to run a callback.
return m_worker.pop_task_result();
2015-10-22 18:06:11 -07:00
} catch (std::exception &exception) {
2016-04-12 14:42:05 -07:00
return {{"error", exception.what()}};
2015-10-22 18:06:11 -07:00
}
2015-10-08 11:57:07 -06:00
}
2015-10-22 10:44:10 -07:00
RPCObjectID RPCServer::store_object(JSObjectRef object) {
2015-10-08 11:57:07 -06:00
static RPCObjectID s_next_id = 1;
2015-10-08 11:57:07 -06:00
RPCObjectID next_id = s_next_id++;
m_objects.emplace(next_id, js::Protected<JSObjectRef>(m_context, object));
2015-10-08 11:57:07 -06:00
return next_id;
}
2016-04-12 14:42:05 -07:00
json RPCServer::serialize_json_value(JSValueRef js_value) {
switch (JSValueGetType(m_context, js_value)) {
2015-10-08 11:57:07 -06:00
case kJSTypeUndefined:
2015-10-22 16:49:32 -07:00
return json::object();
2015-10-08 11:57:07 -06:00
case kJSTypeNull:
2015-10-22 16:49:32 -07:00
return {{"value", json(nullptr)}};
2015-10-08 11:57:07 -06:00
case kJSTypeBoolean:
2016-04-12 14:42:05 -07:00
return {{"value", jsc::Value::to_boolean(m_context, js_value)}};
2015-10-08 11:57:07 -06:00
case kJSTypeNumber:
2016-04-12 14:42:05 -07:00
return {{"value", jsc::Value::to_number(m_context, js_value)}};
2015-10-08 11:57:07 -06:00
case kJSTypeString:
2016-04-12 14:42:05 -07:00
return {{"value", jsc::Value::to_string(m_context, js_value)}};
2015-10-08 11:57:07 -06:00
case kJSTypeObject:
break;
}
2016-04-12 14:42:05 -07:00
JSObjectRef js_object = jsc::Value::validated_to_object(m_context, js_value);
2015-10-08 11:57:07 -06:00
if (jsc::Object::is_instance<js::RealmObjectClass<jsc::Types>>(m_context, js_object)) {
auto object = jsc::Object::get_internal<js::RealmObjectClass<jsc::Types>>(js_object);
2015-10-22 10:44:10 -07:00
return {
{"type", RealmObjectTypesObject},
2015-10-22 17:59:05 -07:00
{"id", store_object(js_object)},
2016-01-04 18:13:09 -08:00
{"schema", serialize_object_schema(object->get_object_schema())}
2015-10-08 11:57:07 -06:00
};
}
else if (jsc::Object::is_instance<js::ListClass<jsc::Types>>(m_context, js_object)) {
auto list = jsc::Object::get_internal<js::ListClass<jsc::Types>>(js_object);
2015-10-22 10:44:10 -07:00
return {
{"type", RealmObjectTypesList},
2015-10-22 17:59:05 -07:00
{"id", store_object(js_object)},
{"size", list->size()},
2016-01-04 18:13:09 -08:00
{"schema", serialize_object_schema(list->get_object_schema())}
2015-10-08 11:57:07 -06:00
};
}
else if (jsc::Object::is_instance<js::ResultsClass<jsc::Types>>(m_context, js_object)) {
auto results = jsc::Object::get_internal<js::ResultsClass<jsc::Types>>(js_object);
2015-10-22 10:44:10 -07:00
return {
2015-10-22 16:49:32 -07:00
{"type", RealmObjectTypesResults},
2015-10-22 17:59:05 -07:00
{"id", store_object(js_object)},
2015-10-22 16:49:32 -07:00
{"size", results->size()},
2016-01-04 18:13:09 -08:00
{"schema", serialize_object_schema(results->get_object_schema())}
2015-10-08 11:57:07 -06:00
};
}
else if (jsc::Object::is_instance<js::RealmClass<jsc::Types>>(m_context, js_object)) {
return {
{"type", RealmObjectTypesRealm},
{"id", store_object(js_object)},
};
}
2016-04-12 14:42:05 -07:00
else if (jsc::Value::is_array(m_context, js_object)) {
uint32_t length = jsc::Object::validated_get_length(m_context, js_object);
2015-10-22 10:44:10 -07:00
std::vector<json> array;
2016-04-12 14:42:05 -07:00
for (uint32_t i = 0; i < length; i++) {
array.push_back(serialize_json_value(jsc::Object::get_property(m_context, js_object, i)));
2015-10-08 11:57:07 -06:00
}
2015-10-22 16:49:32 -07:00
return {{"value", array}};
2015-10-08 11:57:07 -06:00
}
2016-04-12 14:42:05 -07:00
else if (jsc::Value::is_array_buffer(m_context, js_object)) {
std::string data = Accessor::to_binary(m_context, js_value);
return {
{"type", RealmObjectTypesData},
{"value", base64_encode((unsigned char *)data.data(), data.size())},
};
}
2016-04-12 14:42:05 -07:00
else if (jsc::Value::is_date(m_context, js_object)) {
2015-10-22 10:44:10 -07:00
return {
{"type", RealmObjectTypesDate},
2016-04-12 14:42:05 -07:00
{"value", jsc::Value::to_number(m_context, js_object)},
2015-10-19 16:46:56 -07:00
};
}
else {
// Serialize this JS object as a plain object since it doesn't match any known types above.
2016-04-12 14:42:05 -07:00
std::vector<jsc::String> js_keys = jsc::Object::get_property_names(m_context, js_object);
std::vector<std::string> keys;
std::vector<json> values;
2016-04-12 14:42:05 -07:00
for (auto &js_key : js_keys) {
JSValueRef js_value = jsc::Object::get_property(m_context, js_object, js_key);
2016-04-12 14:42:05 -07:00
keys.push_back(js_key);
values.push_back(serialize_json_value(js_value));
}
return {
{"type", RealmObjectTypesDictionary},
{"keys", keys},
{"values", values},
};
}
2015-10-22 10:44:10 -07:00
assert(0);
2015-10-08 11:57:07 -06:00
}
2015-10-26 15:27:43 -07:00
json RPCServer::serialize_object_schema(const realm::ObjectSchema &object_schema) {
std::vector<std::string> properties;
2016-04-12 14:42:05 -07:00
for (auto &prop : object_schema.properties) {
properties.push_back(prop.name);
2015-10-08 11:57:07 -06:00
}
2015-10-22 10:44:10 -07:00
return {
2015-10-22 17:59:05 -07:00
{"name", object_schema.name},
2015-10-22 16:49:32 -07:00
{"properties", properties},
2015-10-08 11:57:07 -06:00
};
}
2016-04-12 14:42:05 -07:00
JSValueRef RPCServer::deserialize_json_value(const json dict) {
2015-10-22 16:49:32 -07:00
json oid = dict["id"];
if (oid.is_number()) {
2015-10-22 17:59:05 -07:00
return m_objects[oid.get<RPCObjectID>()];
2015-10-08 11:57:07 -06:00
}
2015-10-22 15:31:26 -07:00
json value = dict["value"];
2015-10-22 16:49:32 -07:00
json type = dict["type"];
2016-04-12 14:42:05 -07:00
2015-10-22 16:49:32 -07:00
if (type.is_string()) {
std::string type_string = type.get<std::string>();
2016-04-12 14:42:05 -07:00
2015-10-22 16:49:32 -07:00
if (type_string == RealmObjectTypesFunction) {
RPCObjectID callback_id = value.get<RPCObjectID>();
if (!m_callbacks.count(callback_id)) {
JSObjectRef callback = JSObjectMakeFunctionWithCallback(m_context, nullptr, js::wrap<run_callback>);
// Curry the first argument to be the callback id.
JSValueRef bind_args[2] = {jsc::Value::from_null(m_context), jsc::Value::from_number(m_context, callback_id)};
JSValueRef bound_callback = jsc::Object::call_method(m_context, callback, "bind", 2, bind_args);
callback = jsc::Value::to_function(m_context, bound_callback);
m_callbacks.emplace(callback_id, js::Protected<JSObjectRef>(m_context, callback));
}
return m_callbacks.at(callback_id);
2015-10-22 16:49:32 -07:00
}
else if (type_string == RealmObjectTypesDictionary) {
2016-04-12 14:42:05 -07:00
JSObjectRef js_object = jsc::Object::create_empty(m_context);
json keys = dict["keys"];
json values = dict["values"];
size_t count = keys.size();
for (size_t i = 0; i < count; i++) {
2016-04-12 14:42:05 -07:00
std::string js_key = keys.at(i);
JSValueRef js_value = deserialize_json_value(values.at(i));
2016-04-12 14:42:05 -07:00
jsc::Object::set_property(m_context, js_object, js_key, js_value);
}
return js_object;
}
else if (type_string == RealmObjectTypesData) {
std::string bytes;
if (!base64_decode(value.get<std::string>(), &bytes)) {
throw std::runtime_error("Failed to decode base64 encoded data");
}
2016-04-12 14:42:05 -07:00
return Accessor::from_binary(m_context, realm::BinaryData(bytes));
}
else if (type_string == RealmObjectTypesDate) {
2016-04-12 14:42:05 -07:00
return jsc::Object::create_date(m_context, value.get<double>());
2015-10-19 16:46:56 -07:00
}
else if (type_string == RealmObjectTypesUndefined) {
2016-04-12 14:42:05 -07:00
return jsc::Value::from_undefined(m_context);
}
assert(0);
2015-10-19 16:46:56 -07:00
}
2015-10-22 16:49:32 -07:00
if (value.is_null()) {
2016-04-12 14:42:05 -07:00
return jsc::Value::from_null(m_context);
2015-10-08 11:57:07 -06:00
}
2015-10-22 15:31:26 -07:00
else if (value.is_boolean()) {
2016-04-12 14:42:05 -07:00
return jsc::Value::from_boolean(m_context, value.get<bool>());
2015-10-08 11:57:07 -06:00
}
2015-10-22 15:31:26 -07:00
else if (value.is_number()) {
2016-04-12 14:42:05 -07:00
return jsc::Value::from_number(m_context, value.get<double>());
2015-10-08 11:57:07 -06:00
}
2015-10-22 15:31:26 -07:00
else if (value.is_string()) {
2016-04-12 14:42:05 -07:00
return jsc::Value::from_string(m_context, value.get<std::string>());
2015-10-08 11:57:07 -06:00
}
2015-10-22 15:31:26 -07:00
else if (value.is_array()) {
size_t count = value.size();
2015-10-22 17:59:05 -07:00
JSValueRef js_values[count];
2015-10-08 11:57:07 -06:00
2015-10-22 15:31:26 -07:00
for (size_t i = 0; i < count; i++) {
2015-10-22 17:59:05 -07:00
js_values[i] = deserialize_json_value(value.at(i));
2015-10-08 11:57:07 -06:00
}
2016-04-12 14:42:05 -07:00
return jsc::Object::create_array(m_context, (uint32_t)count, js_values);
2015-10-08 11:57:07 -06:00
}
assert(0);
2015-10-08 11:57:07 -06:00
}