Merge branch 'master' of github.com:realm/realm-js into kneth/sync-2.0.0-rc28
This commit is contained in:
commit
2e08da3cb8
|
@ -8,7 +8,8 @@ X.Y.Z Release notes
|
||||||
* Added support for aggregate functions on `Realm.Results` and `Realm.List` of primitive types.
|
* Added support for aggregate functions on `Realm.Results` and `Realm.List` of primitive types.
|
||||||
|
|
||||||
### Bug fixes
|
### Bug fixes
|
||||||
* None
|
* Avoid closing then reopening a sync session when using `Realm.open` (#1391).
|
||||||
|
* Respect custom Realm paths when using `Realm.open` (#1392 and #1393).
|
||||||
|
|
||||||
### Internal
|
### Internal
|
||||||
* Upgrading to Realm Sync 2.0.0-rc28.
|
* Upgrading to Realm Sync 2.0.0-rc28.
|
||||||
|
|
|
@ -128,6 +128,7 @@ util.createMethods(Realm.prototype, objectTypes.REALM, [
|
||||||
'removeListener',
|
'removeListener',
|
||||||
'removeAllListeners',
|
'removeAllListeners',
|
||||||
'close',
|
'close',
|
||||||
|
'_waitForDownload',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// Mutating methods:
|
// Mutating methods:
|
||||||
|
@ -189,12 +190,6 @@ Object.defineProperties(Realm, {
|
||||||
rpc.clearTestState();
|
rpc.clearTestState();
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
_waitForDownload: {
|
|
||||||
value: function(_config, sessionCallback, callback) {
|
|
||||||
sessionCallback();
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
for (let i = 0, len = debugHosts.length; i < len; i++) {
|
for (let i = 0, len = debugHosts.length; i < len; i++) {
|
||||||
|
|
|
@ -31,6 +31,25 @@ function setConstructorOnPrototype(klass) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Return a configuration usable by `Realm.open` when waiting for a download.
|
||||||
|
// It must have caching disabled, and no schema or schema version specified.
|
||||||
|
function waitForDownloadConfig(config) {
|
||||||
|
if (!config) {
|
||||||
|
return {_cache: false};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof config == 'string') {
|
||||||
|
return {path: config, _cache: false};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof config == 'object') {
|
||||||
|
return Object.assign({}, config, {schema: undefined, schemaVersion: undefined, _cache: false});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unknown type. Pass the config through.
|
||||||
|
return config;
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = function(realmConstructor) {
|
module.exports = function(realmConstructor) {
|
||||||
// Add the specified Array methods to the Collection prototype.
|
// Add the specified Array methods to the Collection prototype.
|
||||||
Object.defineProperties(realmConstructor.Collection.prototype, require('./collection-methods'));
|
Object.defineProperties(realmConstructor.Collection.prototype, require('./collection-methods'));
|
||||||
|
@ -45,7 +64,8 @@ module.exports = function(realmConstructor) {
|
||||||
open(config) {
|
open(config) {
|
||||||
let syncSession;
|
let syncSession;
|
||||||
let promise = new Promise((resolve, reject) => {
|
let promise = new Promise((resolve, reject) => {
|
||||||
realmConstructor._waitForDownload(config,
|
let realm = new realmConstructor(waitForDownloadConfig(config));
|
||||||
|
realm._waitForDownload(
|
||||||
(session) => {
|
(session) => {
|
||||||
syncSession = session;
|
syncSession = session;
|
||||||
},
|
},
|
||||||
|
@ -55,7 +75,7 @@ module.exports = function(realmConstructor) {
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
try {
|
try {
|
||||||
let syncedRealm = new this(config);
|
let syncedRealm = new realmConstructor(config);
|
||||||
//FIXME: RN hangs here. Remove when node's makeCallback alternative is implemented
|
//FIXME: RN hangs here. Remove when node's makeCallback alternative is implemented
|
||||||
setTimeout(() => { resolve(syncedRealm); }, 1);
|
setTimeout(() => { resolve(syncedRealm); }, 1);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
@ -80,25 +100,15 @@ module.exports = function(realmConstructor) {
|
||||||
const message = "Realm.openAsync is now deprecated in favor of Realm.open. This function will be removed in future versions.";
|
const message = "Realm.openAsync is now deprecated in favor of Realm.open. This function will be removed in future versions.";
|
||||||
(console.warn || console.log).call(console, message);
|
(console.warn || console.log).call(console, message);
|
||||||
|
|
||||||
realmConstructor._waitForDownload(config,
|
let promise = this.open(config)
|
||||||
(syncSession) => {
|
|
||||||
if (progressCallback) {
|
if (progressCallback) {
|
||||||
syncSession.addProgressNotification('download', 'forCurrentlyOutstandingWork', progressCallback);
|
promise.progress(progressCallback)
|
||||||
}
|
|
||||||
},
|
|
||||||
(error) => {
|
|
||||||
if (error) {
|
|
||||||
setTimeout(() => { callback(error); }, 1);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
try {
|
|
||||||
let syncedRealm = new this(config);
|
|
||||||
//FIXME: RN hangs here. Remove when node's makeCallback alternative is implemented
|
|
||||||
setTimeout(() => { callback(null, syncedRealm); }, 1);
|
|
||||||
} catch (e) {
|
|
||||||
setTimeout(() => { callback(e); }, 1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
promise.then(realm => {
|
||||||
|
callback(null, realm)
|
||||||
|
}).catch(error => {
|
||||||
|
callback(error);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
|
|
|
@ -72,31 +72,11 @@ function getSpecialPurposeRealm(user, realmName, schema) {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const _Realm = user.constructor._realmConstructor;
|
let Realm = user.constructor._realmConstructor;
|
||||||
|
return Realm.open(config).then(realm => {
|
||||||
return new Promise((resolve, reject) => {
|
user[specialPurposeRealmsKey][realmName] = realm;
|
||||||
_Realm._waitForDownload(config, (error) => {
|
return realm;
|
||||||
|
})
|
||||||
// FIXME: I don't understand why, but removing the following setTimeout causes the subsequent
|
|
||||||
// setTimeout call (when resolving the promise) to hang on RN iOS.
|
|
||||||
// This might be related to our general makeCallback issue: #1255.
|
|
||||||
setTimeout(() => {}, 1);
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
setTimeout(() => reject(error), 1);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
try {
|
|
||||||
let syncedRealm = new _Realm(config);
|
|
||||||
user[specialPurposeRealmsKey][realmName] = syncedRealm;
|
|
||||||
//FIXME: RN hangs here. Remove when node's makeCallback alternative is implemented (#1255)
|
|
||||||
setTimeout(() => resolve(syncedRealm), 1);
|
|
||||||
} catch (e) {
|
|
||||||
setTimeout(() => reject(e), 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function createInManagementRealm(user, modelName, modelInitializer) {
|
function createInManagementRealm(user, modelName, modelInitializer) {
|
||||||
|
|
|
@ -218,7 +218,6 @@ public:
|
||||||
{"clearTestState", wrap<clear_test_state>},
|
{"clearTestState", wrap<clear_test_state>},
|
||||||
{"copyBundledRealmFiles", wrap<copy_bundled_realm_files>},
|
{"copyBundledRealmFiles", wrap<copy_bundled_realm_files>},
|
||||||
{"deleteFile", wrap<delete_file>},
|
{"deleteFile", wrap<delete_file>},
|
||||||
{"_waitForDownload", wrap<wait_for_download_completion>},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
PropertyMap<T> const static_properties = {
|
PropertyMap<T> const static_properties = {
|
||||||
|
@ -241,6 +240,7 @@ public:
|
||||||
{"close", wrap<close>},
|
{"close", wrap<close>},
|
||||||
{"compact", wrap<compact>},
|
{"compact", wrap<compact>},
|
||||||
{"deleteModel", wrap<delete_model>},
|
{"deleteModel", wrap<delete_model>},
|
||||||
|
{"_waitForDownload", wrap<wait_for_download_completion>},
|
||||||
#if REALM_ENABLE_SYNC
|
#if REALM_ENABLE_SYNC
|
||||||
{"_subscribeToObjects", wrap<subscribe_to_objects>},
|
{"_subscribeToObjects", wrap<subscribe_to_objects>},
|
||||||
#endif
|
#endif
|
||||||
|
@ -503,6 +503,12 @@ void RealmClass<T>::constructor(ContextType ctx, ObjectType this_object, size_t
|
||||||
realm_ptr->reset();
|
realm_ptr->reset();
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const String cache_string = "_cache";
|
||||||
|
ValueType cache_value = Object::get_property(ctx, object, cache_string);
|
||||||
|
if (!Value::is_undefined(ctx, cache_value)) {
|
||||||
|
config.cache = Value::validated_to_boolean(ctx, cache_value, "_cache");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -694,30 +700,17 @@ void RealmClass<T>::get_sync_session(ContextType ctx, ObjectType object, ReturnV
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
void RealmClass<T>::wait_for_download_completion(ContextType ctx, ObjectType this_object, Arguments args, ReturnValue &return_value) {
|
void RealmClass<T>::wait_for_download_completion(ContextType ctx, ObjectType this_object, Arguments args, ReturnValue &return_value) {
|
||||||
args.validate_maximum(3);
|
args.validate_maximum(2);
|
||||||
auto config_object = Value::validated_to_object(ctx, args[0]);
|
auto callback_function = Value::validated_to_function(ctx, args[0 + (args.count == 2)]);
|
||||||
auto callback_function = Value::validated_to_function(ctx, args[1 + (args.count == 3)]);
|
|
||||||
|
|
||||||
ValueType session_callback = Value::from_null(ctx);
|
ValueType session_callback = Value::from_null(ctx);
|
||||||
if (args.count == 3) {
|
if (args.count == 2) {
|
||||||
session_callback = Value::validated_to_function(ctx, args[1]);
|
session_callback = Value::validated_to_function(ctx, args[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if REALM_ENABLE_SYNC
|
#if REALM_ENABLE_SYNC
|
||||||
ValueType sync_config_value = Object::get_property(ctx, config_object, "sync");
|
auto realm = *get_internal<T, RealmClass<T>>(this_object);
|
||||||
if (!Value::is_undefined(ctx, sync_config_value)) {
|
if (auto* sync_config = realm->config().sync_config.get()) {
|
||||||
realm::Realm::Config config;
|
|
||||||
config.cache = false;
|
|
||||||
static const String encryption_key_string = "encryptionKey";
|
|
||||||
ValueType encryption_key_value = Object::get_property(ctx, config_object, encryption_key_string);
|
|
||||||
if (!Value::is_undefined(ctx, encryption_key_value)) {
|
|
||||||
auto encryption_key = Value::validated_to_binary(ctx, encryption_key_value, "encryptionKey");
|
|
||||||
config.encryption_key.assign(encryption_key.data(), encryption_key.data() + encryption_key.size());
|
|
||||||
}
|
|
||||||
|
|
||||||
Protected<ObjectType> thiz(ctx, this_object);
|
|
||||||
SyncClass<T>::populate_sync_config(ctx, thiz, config_object, config);
|
|
||||||
|
|
||||||
Protected<FunctionType> protected_callback(ctx, callback_function);
|
Protected<FunctionType> protected_callback(ctx, callback_function);
|
||||||
Protected<ObjectType> protected_this(ctx, this_object);
|
Protected<ObjectType> protected_this(ctx, this_object);
|
||||||
Protected<typename T::GlobalContext> protected_ctx(Context<T>::get_global_context(ctx));
|
Protected<typename T::GlobalContext> protected_ctx(Context<T>::get_global_context(ctx));
|
||||||
|
@ -738,52 +731,16 @@ void RealmClass<T>::wait_for_download_completion(ContextType ctx, ObjectType thi
|
||||||
callback_arguments[0] = object;
|
callback_arguments[0] = object;
|
||||||
Function<T>::callback(protected_ctx, protected_callback, protected_this, 1, callback_arguments);
|
Function<T>::callback(protected_ctx, protected_callback, protected_this, 1, callback_arguments);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We keep our Realm instance alive until the callback has had a chance to open its own instance.
|
||||||
|
// This allows it to share the sync session that our Realm opened.
|
||||||
|
if (realm)
|
||||||
|
realm->close();
|
||||||
});
|
});
|
||||||
std::function<WaitHandler> waitFunc = std::move(wait_handler);
|
|
||||||
|
|
||||||
std::function<ProgressHandler> progressFunc;
|
|
||||||
|
|
||||||
SharedRealm realm;
|
|
||||||
try {
|
|
||||||
realm = realm::Realm::get_shared_realm(config);
|
|
||||||
}
|
|
||||||
catch (const RealmFileException& ex) {
|
|
||||||
handleRealmFileException(ctx, config, ex);
|
|
||||||
}
|
|
||||||
catch (...) {
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (auto sync_config = config.sync_config)
|
|
||||||
{
|
|
||||||
static const String progressFuncName = "_onDownloadProgress";
|
|
||||||
bool progressFuncDefined = false;
|
|
||||||
if (!Value::is_boolean(ctx, sync_config_value) && !Value::is_undefined(ctx, sync_config_value))
|
|
||||||
{
|
|
||||||
auto sync_config_object = Value::validated_to_object(ctx, sync_config_value);
|
|
||||||
|
|
||||||
ValueType progressFuncValue = Object::get_property(ctx, sync_config_object, progressFuncName);
|
|
||||||
progressFuncDefined = !Value::is_undefined(ctx, progressFuncValue);
|
|
||||||
|
|
||||||
if (progressFuncDefined)
|
|
||||||
{
|
|
||||||
Protected<FunctionType> protected_progressCallback(protected_ctx, Value::validated_to_function(protected_ctx, progressFuncValue));
|
|
||||||
EventLoopDispatcher<ProgressHandler> 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<T>::callback(protected_ctx, protected_progressCallback, protected_this, 2, callback_arguments);
|
|
||||||
});
|
|
||||||
|
|
||||||
progressFunc = std::move(progress_handler);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
std::shared_ptr<SyncUser> user = sync_config->user;
|
std::shared_ptr<SyncUser> user = sync_config->user;
|
||||||
if (user && user->state() != SyncUser::State::Error) {
|
if (user && user->state() != SyncUser::State::Error) {
|
||||||
if (auto session = user->session_for_on_disk_path(config.path)) {
|
if (auto session = user->session_for_on_disk_path(realm->config().path)) {
|
||||||
if (!Value::is_null(ctx, session_callback)) {
|
if (!Value::is_null(ctx, session_callback)) {
|
||||||
FunctionType session_callback_func = Value::to_function(ctx, session_callback);
|
FunctionType session_callback_func = Value::to_function(ctx, session_callback);
|
||||||
auto syncSession = create_object<T, SessionClass<T>>(ctx, new WeakSession(session));
|
auto syncSession = create_object<T, SessionClass<T>>(ctx, new WeakSession(session));
|
||||||
|
@ -792,14 +749,7 @@ void RealmClass<T>::wait_for_download_completion(ContextType ctx, ObjectType thi
|
||||||
Function<T>::callback(protected_ctx, session_callback_func, protected_this, 1, callback_arguments);
|
Function<T>::callback(protected_ctx, session_callback_func, protected_this, 1, callback_arguments);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (progressFuncDefined) {
|
session->wait_for_download_completion(std::move(wait_handler));
|
||||||
session->register_progress_notifier(std::move(progressFunc), SyncSession::NotifierType::download, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
session->wait_for_download_completion([=](std::error_code error_code) {
|
|
||||||
realm->close(); //capture and keep realm instance for until here
|
|
||||||
waitFunc(error_code);
|
|
||||||
});
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -814,9 +764,6 @@ void RealmClass<T>::wait_for_download_completion(ContextType ctx, ObjectType thi
|
||||||
Function<T>::callback(protected_ctx, protected_callback, protected_this, 1, callback_arguments);
|
Function<T>::callback(protected_ctx, protected_callback, protected_this, 1, callback_arguments);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
#else
|
|
||||||
static_cast<void>(config_object);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
Function<T>::callback(ctx, callback_function, this_object, 0, nullptr);
|
Function<T>::callback(ctx, callback_function, this_object, 0, nullptr);
|
||||||
|
|
|
@ -249,89 +249,6 @@ module.exports = {
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
testProgressNotificationsForRealmOpen() {
|
|
||||||
if (!isNodeProccess) {
|
|
||||||
return Promise.resolve();
|
|
||||||
}
|
|
||||||
|
|
||||||
const username = uuid();
|
|
||||||
const realmName = uuid();
|
|
||||||
const expectedObjectsCount = 3;
|
|
||||||
|
|
||||||
return runOutOfProcess(__dirname + '/download-api-helper.js', username, realmName, REALM_MODULE_PATH)
|
|
||||||
.then(() => {
|
|
||||||
return Realm.Sync.User.login('http://localhost:9080', username, 'password').then(user => {
|
|
||||||
const accessTokenRefreshed = this;
|
|
||||||
let successCounter = 0;
|
|
||||||
let progressNotificationCalled = false;
|
|
||||||
let config = {
|
|
||||||
sync: {
|
|
||||||
user,
|
|
||||||
url: `realm://localhost:9080/~/${realmName}`,
|
|
||||||
_onDownloadProgress: (transferred, total) => {
|
|
||||||
progressNotificationCalled = true
|
|
||||||
},
|
|
||||||
},
|
|
||||||
schema: [{ name: 'Dog', properties: { name: 'string' } }],
|
|
||||||
};
|
|
||||||
|
|
||||||
return Realm.open(config)
|
|
||||||
.then(realm => {
|
|
||||||
return realm.syncSession;
|
|
||||||
}).then(session => {
|
|
||||||
TestCase.assertTrue(progressNotificationCalled, "Progress notification not called for Realm.open");
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
testProgressNotificationsForRealmOpenAsync() {
|
|
||||||
if (!isNodeProccess) {
|
|
||||||
return Promise.resolve();
|
|
||||||
}
|
|
||||||
|
|
||||||
const username = uuid();
|
|
||||||
const realmName = uuid();
|
|
||||||
const expectedObjectsCount = 3;
|
|
||||||
|
|
||||||
return runOutOfProcess(__dirname + '/download-api-helper.js', username, realmName, REALM_MODULE_PATH)
|
|
||||||
.then(() => {
|
|
||||||
return Realm.Sync.User.login('http://localhost:9080', username, 'password').then(user => {
|
|
||||||
return new Promise((resolve, reject) => {
|
|
||||||
let progressNotificationCalled = false;
|
|
||||||
let config = {
|
|
||||||
sync: { user, url: `realm://localhost:9080/~/${realmName}`,
|
|
||||||
_onDownloadProgress: (transferred, total) => {
|
|
||||||
progressNotificationCalled = true
|
|
||||||
},
|
|
||||||
},
|
|
||||||
schema: [{ name: 'Dog', properties: { name: 'string' } }],
|
|
||||||
};
|
|
||||||
|
|
||||||
Realm.openAsync(config, (error, realm) => {
|
|
||||||
try {
|
|
||||||
if (error) {
|
|
||||||
reject(error);
|
|
||||||
}
|
|
||||||
|
|
||||||
setTimeout(() => {
|
|
||||||
try {
|
|
||||||
TestCase.assertTrue(progressNotificationCalled, "Progress notification not called for Realm.openAsync");
|
|
||||||
resolve();
|
|
||||||
} catch (e) {
|
|
||||||
reject(e);
|
|
||||||
}
|
|
||||||
}, 50);
|
|
||||||
}
|
|
||||||
catch (e) {
|
|
||||||
reject(e);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
testRealmOpenAsyncNoSchema() {
|
testRealmOpenAsyncNoSchema() {
|
||||||
if (!isNodeProccess) {
|
if (!isNodeProccess) {
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
|
@ -697,7 +614,7 @@ module.exports = {
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
testProgressNotificationsForRealmOpen2() {
|
testProgressNotificationsForRealmOpen() {
|
||||||
if (!isNodeProccess) {
|
if (!isNodeProccess) {
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
|
@ -736,7 +653,7 @@ module.exports = {
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
testProgressNotificationsForRealmOpenAsync2() {
|
testProgressNotificationsForRealmOpenAsync() {
|
||||||
if (!isNodeProccess) {
|
if (!isNodeProccess) {
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue