//////////////////////////////////////////////////////////////////////////// // // 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 #include #include #include #include #include #include "event_loop_dispatcher.hpp" #include "platform.hpp" #include "js_class.hpp" #include "js_collection.hpp" #include "sync/sync_manager.hpp" #include "sync/sync_config.hpp" #include "sync/sync_session.hpp" #include "sync/sync_user.hpp" #include "realm/util/logger.hpp" #include "realm/util/uri.hpp" namespace realm { namespace js { using SharedUser = std::shared_ptr; using WeakSession = std::weak_ptr; template class UserClass : public ClassDefinition { using GlobalContextType = typename T::GlobalContext; using ContextType = typename T::Context; using FunctionType = typename T::Function; using ObjectType = typename T::Object; using ValueType = typename T::Value; using String = js::String; using Object = js::Object; using Value = js::Value; using Function = js::Function; using ReturnValue = js::ReturnValue; public: std::string const name = "User"; static FunctionType create_constructor(ContextType); static void get_server(ContextType, ObjectType, ReturnValue &); static void get_identity(ContextType, ObjectType, ReturnValue &); static void get_token(ContextType, ObjectType, ReturnValue &); static void is_admin(ContextType, ObjectType, ReturnValue &); PropertyMap const properties = { {"server", {wrap, nullptr}}, {"identity", {wrap, nullptr}}, {"token", {wrap, nullptr}}, {"isAdmin", {wrap, nullptr}}, }; static void create_user(ContextType, FunctionType, ObjectType, size_t, const ValueType[], ReturnValue &); static void admin_user(ContextType, FunctionType, ObjectType, size_t, const ValueType[], ReturnValue &); MethodMap const static_methods = { {"createUser", wrap}, {"_adminUser", wrap} }; /*static void current_user(ContextType ctx, ObjectType object, ReturnValue &return_value);*/ static void all_users(ContextType ctx, ObjectType object, ReturnValue &return_value); PropertyMap const static_properties = { /*{"current", {wrap, nullptr}},*/ {"all", {wrap, nullptr}}, }; static void logout(ContextType, FunctionType, ObjectType, size_t, const ValueType[], ReturnValue &); static void session_for_on_disk_path(ContextType, FunctionType, ObjectType, size_t, const ValueType[], ReturnValue &); MethodMap const methods = { {"logout", wrap}, {"_sessionForOnDiskPath", wrap} }; }; template void UserClass::get_server(ContextType ctx, ObjectType object, ReturnValue &return_value) { std::string server = get_internal>(object)->get()->server_url(); return_value.set(server); } template void UserClass::get_identity(ContextType ctx, ObjectType object, ReturnValue &return_value) { std::string identity = get_internal>(object)->get()->identity(); return_value.set(identity); } template void UserClass::get_token(ContextType ctx, ObjectType object, ReturnValue &return_value) { std::string token = get_internal>(object)->get()->refresh_token(); return_value.set(token); } template void UserClass::is_admin(ContextType ctx, ObjectType object, ReturnValue &return_value) { return_value.set(get_internal>(object)->get()->is_admin()); } template void UserClass::create_user(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) { validate_argument_count(argc, 3, 5); SyncUserIdentifier userIdentifier { Value::validated_to_string(ctx, arguments[1], "identity"), Value::validated_to_string(ctx, arguments[0], "authServerUrl") }; SharedUser *user = new SharedUser(SyncManager::shared().get_user( userIdentifier, Value::validated_to_string(ctx, arguments[2], "refreshToken") )); if (argc == 5) { (*user)->set_is_admin(Value::validated_to_boolean(ctx, arguments[4], "isAdmin")); } return_value.set(create_object>(ctx, user)); } template void UserClass::admin_user(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) { validate_argument_count(argc, 2, 2); SharedUser *user = new SharedUser(SyncManager::shared().get_admin_token_user( Value::validated_to_string(ctx, arguments[0], "authServerUrl"), Value::validated_to_string(ctx, arguments[1], "refreshToken") )); return_value.set(create_object>(ctx, user)); } template void UserClass::all_users(ContextType ctx, ObjectType object, ReturnValue &return_value) { auto users = Object::create_empty(ctx); for (auto user : SyncManager::shared().all_logged_in_users()) { if (user->token_type() == SyncUser::TokenType::Normal) { Object::set_property(ctx, users, user->identity(), create_object>(ctx, new SharedUser(user)), ReadOnly | DontDelete); } } return_value.set(users); } template void UserClass::logout(ContextType ctx, FunctionType, ObjectType this_object, size_t, const ValueType[], ReturnValue &) { get_internal>(this_object)->get()->log_out(); } template class SessionClass : public ClassDefinition { using ContextType = typename T::Context; using FunctionType = typename T::Function; using ObjectType = typename T::Object; using ValueType = typename T::Value; using String = js::String; using Object = js::Object; using Value = js::Value; using ReturnValue = js::ReturnValue; public: std::string const name = "Session"; using ProgressHandler = void(uint64_t transferred_bytes, uint64_t transferrable_bytes); static FunctionType create_constructor(ContextType); static void get_config(ContextType, ObjectType, ReturnValue &); static void get_user(ContextType, ObjectType, ReturnValue &); static void get_url(ContextType, ObjectType, ReturnValue &); static void get_state(ContextType, ObjectType, ReturnValue &); static void simulate_error(ContextType, FunctionType, ObjectType, size_t, const ValueType[], ReturnValue &); static void refresh_access_token(ContextType, FunctionType, ObjectType, size_t, const ValueType[], ReturnValue &); static void add_progress_notification(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &); static void remove_progress_notification(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &); PropertyMap const properties = { {"config", {wrap, nullptr}}, {"user", {wrap, nullptr}}, {"url", {wrap, nullptr}}, {"state", {wrap, nullptr}} }; MethodMap const methods = { {"_simulateError", wrap}, {"_refreshAccessToken", wrap}, {"addProgressNotification", wrap}, {"removeProgressNotification", wrap}, }; }; template class SyncSessionErrorHandlerFunctor { public: SyncSessionErrorHandlerFunctor(typename T::Context ctx, typename T::Function error_func) : m_ctx(Context::get_global_context(ctx)) , m_func(ctx, error_func) { } typename T::Function func() const { return m_func; } void operator()(std::shared_ptr session, SyncError error) { HANDLESCOPE auto error_object = Object::create_empty(m_ctx); Object::set_property(m_ctx, error_object, "message", Value::from_string(m_ctx, error.message)); Object::set_property(m_ctx, error_object, "isFatal", Value::from_boolean(m_ctx, error.is_fatal)); Object::set_property(m_ctx, error_object, "category", Value::from_string(m_ctx, error.error_code.category().name())); Object::set_property(m_ctx, error_object, "code", Value::from_number(m_ctx, error.error_code.value())); auto user_info = Object::create_empty(m_ctx); for (auto& kvp : error.user_info) { Object::set_property(m_ctx, user_info, kvp.first, Value::from_string(m_ctx, kvp.second)); } Object::set_property(m_ctx, error_object, "userInfo", user_info); typename T::Value arguments[2]; arguments[0] = create_object>(m_ctx, new WeakSession(session)); arguments[1] = error_object; Function::callback(m_ctx, m_func, typename T::Object(), 2, arguments); } private: const Protected m_ctx; const Protected m_func; }; template class SSLVerifyCallbackSyncThreadFunctor { public: SSLVerifyCallbackSyncThreadFunctor(typename T::Context ctx, typename T::Function ssl_verify_func) : m_ctx(Context::get_global_context(ctx)) , m_func(ctx, ssl_verify_func) , m_event_loop_dispatcher {SSLVerifyCallbackSyncThreadFunctor::main_loop_handler} , m_mutex{new std::mutex} , m_cond_var{new std::condition_variable} { } bool operator ()(const std::string& server_address, sync::Session::port_type server_port, const char* pem_data, size_t pem_size, int preverify_ok, int depth) { const std::string pem_certificate {pem_data, pem_size}; { std::lock_guard lock {*m_mutex}; m_ssl_certificate_callback_done = false; } m_event_loop_dispatcher(this, server_address, server_port, pem_certificate, preverify_ok, depth); bool ssl_certificate_accepted = false; { std::unique_lock lock(*m_mutex); m_cond_var->wait(lock, [this] { return this->m_ssl_certificate_callback_done; }); ssl_certificate_accepted = m_ssl_certificate_accepted; } return ssl_certificate_accepted; } static void main_loop_handler(SSLVerifyCallbackSyncThreadFunctor* this_object, const std::string& server_address, sync::Session::port_type server_port, const std::string& pem_certificate, int preverify_ok, int depth) { HANDLESCOPE const Protected& ctx = this_object->m_ctx; typename T::Object ssl_certificate_object = Object::create_empty(ctx); Object::set_property(ctx, ssl_certificate_object, "serverAddress", Value::from_string(ctx, server_address)); Object::set_property(ctx, ssl_certificate_object, "serverPort", Value::from_number(ctx, double(server_port))); Object::set_property(ctx, ssl_certificate_object, "pemCertificate", Value::from_string(ctx, pem_certificate)); Object::set_property(ctx, ssl_certificate_object, "preverifyOk", Value::from_number(ctx, double(preverify_ok))); Object::set_property(ctx, ssl_certificate_object, "depth", Value::from_number(ctx, double(depth))); const int argc = 1; typename T::Value arguments[argc] = { ssl_certificate_object }; typename T::Value ret_val = Function::callback(ctx, this_object->m_func, typename T::Object(), 1, arguments); bool ret_val_bool = Value::to_boolean(ctx, ret_val); { std::lock_guard lock {*this_object->m_mutex}; this_object->m_ssl_certificate_callback_done = true; this_object->m_ssl_certificate_accepted = ret_val_bool; } this_object->m_cond_var->notify_one(); }; private: const Protected m_ctx; const Protected m_func; EventLoopDispatcher* this_object, const std::string& server_address, sync::Session::port_type server_port, const std::string& pem_certificate, int preverify_ok, int depth)> m_event_loop_dispatcher; bool m_ssl_certificate_callback_done = false; bool m_ssl_certificate_accepted = false; std::shared_ptr m_mutex; std::shared_ptr m_cond_var; }; template void UserClass::session_for_on_disk_path(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) { auto user = *get_internal>(this_object); if (auto session = user->session_for_on_disk_path(Value::validated_to_string(ctx, arguments[0]))) { return_value.set(create_object>(ctx, new WeakSession(session))); } else { return_value.set_undefined(); } } template void SessionClass::get_config(ContextType ctx, ObjectType object, ReturnValue &return_value) { if (auto session = get_internal>(object)->lock()) { ObjectType config = Object::create_empty(ctx); Object::set_property(ctx, config, "user", create_object>(ctx, new SharedUser(session->config().user))); Object::set_property(ctx, config, "url", Value::from_string(ctx, session->config().realm_url)); if (auto* dispatcher = session->config().error_handler.template target>()) { auto& handler = *dispatcher->func().template target>(); Object::set_property(ctx, config, "error", handler.func()); } return_value.set(config); } else { return_value.set_undefined(); } } template void SessionClass::get_user(ContextType ctx, ObjectType object, ReturnValue &return_value) { if (auto session = get_internal>(object)->lock()) { return_value.set(create_object>(ctx, new SharedUser(session->config().user))); } else { return_value.set_undefined(); } } template void SessionClass::get_url(ContextType ctx, ObjectType object, ReturnValue &return_value) { if (auto session = get_internal>(object)->lock()) { if (util::Optional url = session->full_realm_url()) { return_value.set(*url); return; } } return_value.set_undefined(); } template void SessionClass::get_state(ContextType ctx, ObjectType object, ReturnValue &return_value) { static const std::string invalid("invalid"); static const std::string inactive("inactive"); static const std::string active("active"); return_value.set(invalid); if (auto session = get_internal>(object)->lock()) { if (session->state() == SyncSession::PublicState::Inactive) { return_value.set(inactive); } else if (session->state() != SyncSession::PublicState::Error) { return_value.set(active); } } } template void SessionClass::simulate_error(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &) { validate_argument_count(argc, 2); if (auto session = get_internal>(this_object)->lock()) { SyncError error; error.error_code = std::error_code(Value::validated_to_number(ctx, arguments[0]), realm::sync::protocol_error_category()); error.message = Value::validated_to_string(ctx, arguments[1]); SyncSession::OnlyForTesting::handle_error(*session, std::move(error)); } } template void SessionClass::refresh_access_token(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &) { validate_argument_count(argc, 2); if (auto session = get_internal>(this_object)->lock()) { std::string access_token = Value::validated_to_string(ctx, arguments[0], "accessToken"); std::string realm_url = Value::validated_to_string(ctx, arguments[1], "realmUrl"); session->refresh_access_token(std::move(access_token), std::move(realm_url)); } } template void SessionClass::add_progress_notification(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) { validate_argument_count(argc, 3); if (auto session = get_internal>(this_object)->lock()) { std::string direction = Value::validated_to_string(ctx, arguments[0], "direction"); std::string mode = Value::validated_to_string(ctx, arguments[1], "mode"); SyncSession::NotifierType notifierType; if (direction == "download") { notifierType = SyncSession::NotifierType::download; } else if (direction == "upload") { notifierType = SyncSession::NotifierType::upload; } else { throw std::invalid_argument("Invalid argument 'direction'. Only 'download' and 'upload' progress notification directions are supported"); } bool is_streaming = false; if (mode == "reportIndefinitely") { is_streaming = true; } else if (mode == "forCurrentlyOutstandingWork") { is_streaming = false; } else { throw std::invalid_argument("Invalid argument 'mode'. Only 'reportIndefinitely' and 'forCurrentlyOutstandingWork' progress notification modes are supported"); } auto callback_function = Value::validated_to_function(ctx, arguments[2], "callback"); Protected protected_callback(ctx, callback_function); Protected protected_this(ctx, this_object); Protected protected_ctx(Context::get_global_context(ctx)); std::function progressFunc; EventLoopDispatcher progress_handler([=](uint64_t transferred_bytes, uint64_t transferrable_bytes) { HANDLESCOPE ValueType callback_arguments[2]; callback_arguments[0] = Value::from_number(protected_ctx, transferred_bytes); callback_arguments[1] = Value::from_number(protected_ctx, transferrable_bytes); Function::callback(protected_ctx, protected_callback, typename T::Object(), 2, callback_arguments); }); progressFunc = std::move(progress_handler); auto registrationToken = session->register_progress_notifier(std::move(progressFunc), notifierType, false); auto syncSession = create_object>(ctx, new WeakSession(session)); PropertyAttributes attributes = ReadOnly | DontEnum | DontDelete; Object::set_property(ctx, callback_function, "_syncSession", syncSession, attributes); Object::set_property(ctx, callback_function, "_registrationToken", Value::from_number(protected_ctx, registrationToken), attributes); } } template void SessionClass::remove_progress_notification(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) { validate_argument_count(argc, 1); auto callback_function = Value::validated_to_function(ctx, arguments[0], "callback"); auto syncSessionProp = Object::get_property(ctx, callback_function, "_syncSession"); if (Value::is_undefined(ctx, syncSessionProp) || Value::is_null(ctx, syncSessionProp)) { return; } auto syncSession = Value::validated_to_object(ctx, syncSessionProp); auto registrationToken = Object::get_property(ctx, callback_function, "_registrationToken"); if (auto session = get_internal>(syncSession)->lock()) { auto reg = Value::validated_to_number(ctx, registrationToken); session->unregister_progress_notifier(reg); } } template class SyncClass : public ClassDefinition { using GlobalContextType = typename T::GlobalContext; using ContextType = typename T::Context; using FunctionType = typename T::Function; using ObjectType = typename T::Object; using ValueType = typename T::Value; using String = js::String; using Object = js::Object; using Value = js::Value; using Function = js::Function; using ReturnValue = js::ReturnValue; public: std::string const name = "Sync"; static FunctionType create_constructor(ContextType); static void set_sync_log_level(ContextType, FunctionType, ObjectType, size_t, const ValueType[], ReturnValue &); // private static std::function session_bind_callback(ContextType ctx, ObjectType sync_constructor); static void populate_sync_config(ContextType, ObjectType realm_constructor, ObjectType config_object, Realm::Config&); // static properties static void get_is_developer_edition(ContextType, ObjectType, ReturnValue &); MethodMap const static_methods = { {"setLogLevel", wrap}, }; }; template inline typename T::Function SyncClass::create_constructor(ContextType ctx) { FunctionType sync_constructor = ObjectWrap>::create_constructor(ctx); PropertyAttributes attributes = ReadOnly | DontEnum | DontDelete; Object::set_property(ctx, sync_constructor, "User", ObjectWrap>::create_constructor(ctx), attributes); Object::set_property(ctx, sync_constructor, "Session", ObjectWrap>::create_constructor(ctx), attributes); // setup synced realmFile paths ensure_directory_exists_for_file(default_realm_file_directory()); SyncManager::shared().configure_file_system(default_realm_file_directory(), SyncManager::MetadataMode::NoEncryption); return sync_constructor; } template void SyncClass::set_sync_log_level(ContextType ctx, FunctionType, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) { validate_argument_count(argc, 1); std::string log_level = Value::validated_to_string(ctx, arguments[0]); std::istringstream in(log_level); // Throws in.imbue(std::locale::classic()); // Throws in.unsetf(std::ios_base::skipws); util::Logger::Level log_level_2 = util::Logger::Level(); in >> log_level_2; // Throws if (!in || !in.eof()) throw std::runtime_error("Bad log level"); realm::SyncManager::shared().set_log_level(log_level_2); } template std::function SyncClass::session_bind_callback(ContextType ctx, ObjectType sync_constructor) { Protected protected_ctx(Context::get_global_context(ctx)); Protected protected_sync_constructor(ctx, sync_constructor); return EventLoopDispatcher([protected_ctx, protected_sync_constructor](const std::string& path, const realm::SyncConfig& config, std::shared_ptr) { HANDLESCOPE ObjectType user_constructor = Object::validated_get_object(protected_ctx, protected_sync_constructor, "User"); FunctionType refreshAccessToken = Object::validated_get_function(protected_ctx, user_constructor, "_refreshAccessToken"); ValueType arguments[3]; arguments[0] = create_object>(protected_ctx, new SharedUser(config.user)); arguments[1] = Value::from_string(protected_ctx, path); arguments[2] = Value::from_string(protected_ctx, config.realm_url); Function::call(protected_ctx, refreshAccessToken, 3, arguments); }); } template void SyncClass::populate_sync_config(ContextType ctx, ObjectType realm_constructor, ObjectType config_object, Realm::Config& config) { ValueType sync_config_value = Object::get_property(ctx, config_object, "sync"); if (Value::is_boolean(ctx, sync_config_value)) { config.force_sync_history = Value::to_boolean(ctx, sync_config_value); } else if (!Value::is_undefined(ctx, sync_config_value)) { auto sync_config_object = Value::validated_to_object(ctx, sync_config_value); ObjectType sync_constructor = Object::validated_get_object(ctx, realm_constructor, "Sync"); auto bind = session_bind_callback(ctx, sync_constructor); std::function error_handler; ValueType error_func = Object::get_property(ctx, sync_config_object, "error"); if (!Value::is_undefined(ctx, error_func)) { error_handler = EventLoopDispatcher(SyncSessionErrorHandlerFunctor(ctx, Value::validated_to_function(ctx, error_func))); } ObjectType user = Object::validated_get_object(ctx, sync_config_object, "user"); SharedUser shared_user = *get_internal>(user); if (shared_user->state() != SyncUser::State::Active) { throw std::runtime_error("User is no longer valid."); } std::string raw_realm_url = Object::validated_get_string(ctx, sync_config_object, "url"); if (shared_user->token_type() == SyncUser::TokenType::Admin) { static std::regex tilde("/~/"); raw_realm_url = std::regex_replace(raw_realm_url, tilde, "/__auth/"); } bool client_validate_ssl = true; ValueType validate_ssl_temp = Object::get_property(ctx, sync_config_object, "validate_ssl"); if (!Value::is_undefined(ctx, validate_ssl_temp)) { client_validate_ssl = Value::validated_to_boolean(ctx, validate_ssl_temp, "validate_ssl"); } util::Optional ssl_trust_certificate_path; ValueType trust_certificate_path_temp = Object::get_property(ctx, sync_config_object, "ssl_trust_certificate_path"); if (!Value::is_undefined(ctx, trust_certificate_path_temp)) { ssl_trust_certificate_path = std::string(Value::validated_to_string(ctx, trust_certificate_path_temp, "ssl_trust_certificate_path")); } else { ssl_trust_certificate_path = util::none; } std::function ssl_verify_callback; ValueType ssl_verify_func = Object::get_property(ctx, sync_config_object, "ssl_verify_callback"); if (!Value::is_undefined(ctx, ssl_verify_func)) { SSLVerifyCallbackSyncThreadFunctor ssl_verify_functor {ctx, Value::validated_to_function(ctx, ssl_verify_func)}; ssl_verify_callback = std::move(ssl_verify_functor); } // FIXME - use make_shared config.sync_config = std::shared_ptr(new SyncConfig{shared_user, raw_realm_url, SyncSessionStopPolicy::AfterChangesUploaded, std::move(bind), std::move(error_handler), nullptr, util::none, client_validate_ssl, ssl_trust_certificate_path, std::move(ssl_verify_callback)}); config.schema_mode = SchemaMode::Additive; config.path = realm::SyncManager::shared().path_for_realm(*shared_user, raw_realm_url); if (!config.encryption_key.empty()) { config.sync_config->realm_encryption_key = std::array(); std::copy_n(config.encryption_key.begin(), config.sync_config->realm_encryption_key->size(), config.sync_config->realm_encryption_key->begin()); } } } } // js } // realm