mirror of
https://github.com/status-im/realm-js.git
synced 2025-01-09 13:55:49 +00:00
Merge pull request #1580 from realm/mar/realm-open-schema-version
Fix Realm.open complaining about the Realm already being open with a different schema version
This commit is contained in:
commit
34ea280492
15
CHANGELOG.md
15
CHANGELOG.md
@ -1,3 +1,17 @@
|
|||||||
|
X.Y.Z Release notes
|
||||||
|
=============================================================
|
||||||
|
### Breaking changes
|
||||||
|
* None.
|
||||||
|
|
||||||
|
### Enhancements
|
||||||
|
* None.
|
||||||
|
|
||||||
|
### Bug fixes
|
||||||
|
* Fix a bug where `Realm.open` could unexpectedly raise a "Realm at path ... already opened with different schema version" error.
|
||||||
|
|
||||||
|
### Internal
|
||||||
|
* None
|
||||||
|
|
||||||
2.1.1 Release notes (2017-12-15)
|
2.1.1 Release notes (2017-12-15)
|
||||||
=============================================================
|
=============================================================
|
||||||
### Breaking changes
|
### Breaking changes
|
||||||
@ -8,6 +22,7 @@
|
|||||||
|
|
||||||
### Bug fixes
|
### Bug fixes
|
||||||
* [Object Server] Fixed a bug where long reconnection happens when a proxy in front of the sync worker returns one of those.
|
* [Object Server] Fixed a bug where long reconnection happens when a proxy in front of the sync worker returns one of those.
|
||||||
|
* Fix a bug where `Realm.open` could unexpectedly raise a "Realm at path ... already opened with different schema version" error.
|
||||||
|
|
||||||
### Internal
|
### Internal
|
||||||
* Updated to Realm Sync 2.1.10 (see "Bug fixes").
|
* Updated to Realm Sync 2.1.10 (see "Bug fixes").
|
||||||
|
@ -62,16 +62,28 @@ module.exports = function(realmConstructor) {
|
|||||||
//Add async open API
|
//Add async open API
|
||||||
Object.defineProperties(realmConstructor, getOwnPropertyDescriptors({
|
Object.defineProperties(realmConstructor, getOwnPropertyDescriptors({
|
||||||
open(config) {
|
open(config) {
|
||||||
|
// For local Realms we open the Realm and return it in a resolved Promise.
|
||||||
|
if (!("sync" in config)) {
|
||||||
|
let promise = Promise.resolve(new realmConstructor(config));
|
||||||
|
promise.progress = (callback) => { };
|
||||||
|
return promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
// For synced Realms we open the Realm without specifying the schema and then wait until
|
||||||
|
// the Realm has finished its initial sync with the server. We then reopen it with the correct
|
||||||
|
// schema. This avoids writing the schema to a potentially read-only Realm file, which would
|
||||||
|
// result in sync rejecting the writes. `_waitForDownload` ensures that the session is kept
|
||||||
|
// alive until our callback has returned, which prevents it from being torn down and recreated
|
||||||
|
// when we close the schemaless Realm and open it with the correct schema.
|
||||||
let syncSession;
|
let syncSession;
|
||||||
let promise = new Promise((resolve, reject) => {
|
let promise = new Promise((resolve, reject) => {
|
||||||
let realm = new realmConstructor(waitForDownloadConfig(config));
|
let realm = new realmConstructor(waitForDownloadConfig(config));
|
||||||
realm._waitForDownload(
|
realm._waitForDownload(
|
||||||
(session) => {
|
(session) => { syncSession = session; },
|
||||||
syncSession = session;
|
|
||||||
},
|
|
||||||
(error) => {
|
(error) => {
|
||||||
|
realm.close();
|
||||||
if (error) {
|
if (error) {
|
||||||
setTimeout(() => { reject(error); }, 1);
|
setTimeout(() => { reject(error); }, 1);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
try {
|
try {
|
||||||
@ -91,7 +103,6 @@ module.exports = function(realmConstructor) {
|
|||||||
|
|
||||||
return promise;
|
return promise;
|
||||||
};
|
};
|
||||||
|
|
||||||
return promise;
|
return promise;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
113
src/js_realm.hpp
113
src/js_realm.hpp
@ -245,9 +245,9 @@ 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>},
|
|
||||||
{"_objectForObjectId", wrap<object_for_object_id>},
|
{"_objectForObjectId", wrap<object_for_object_id>},
|
||||||
#if REALM_ENABLE_SYNC
|
#if REALM_ENABLE_SYNC
|
||||||
|
{"_waitForDownload", wrap<wait_for_download_completion>},
|
||||||
{"_subscribeToObjects", wrap<subscribe_to_objects>},
|
{"_subscribeToObjects", wrap<subscribe_to_objects>},
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
@ -720,6 +720,7 @@ void RealmClass<T>::get_sync_session(ContextType ctx, ObjectType object, ReturnV
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if REALM_ENABLE_SYNC
|
||||||
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(2);
|
args.validate_maximum(2);
|
||||||
@ -730,67 +731,63 @@ void RealmClass<T>::wait_for_download_completion(ContextType ctx, ObjectType thi
|
|||||||
session_callback = Value::validated_to_function(ctx, args[0]);
|
session_callback = Value::validated_to_function(ctx, args[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if REALM_ENABLE_SYNC
|
|
||||||
auto realm = *get_internal<T, RealmClass<T>>(this_object);
|
auto realm = *get_internal<T, RealmClass<T>>(this_object);
|
||||||
if (auto* sync_config = realm->config().sync_config.get()) {
|
auto* sync_config = realm->config().sync_config.get();
|
||||||
Protected<FunctionType> protected_callback(ctx, callback_function);
|
if (!sync_config) {
|
||||||
Protected<ObjectType> protected_this(ctx, this_object);
|
throw std::logic_error("_waitForDownload can only be used on a synchronized Realm.");
|
||||||
Protected<typename T::GlobalContext> protected_ctx(Context<T>::get_global_context(ctx));
|
|
||||||
|
|
||||||
EventLoopDispatcher<WaitHandler> wait_handler([=](std::error_code error_code) {
|
|
||||||
HANDLESCOPE
|
|
||||||
if (!error_code) {
|
|
||||||
//success
|
|
||||||
Function<T>::callback(protected_ctx, protected_callback, typename T::Object(), 0, nullptr);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
//fail
|
|
||||||
ObjectType object = Object::create_empty(protected_ctx);
|
|
||||||
Object::set_property(protected_ctx, object, "message", Value::from_string(protected_ctx, error_code.message()));
|
|
||||||
Object::set_property(protected_ctx, object, "errorCode", Value::from_number(protected_ctx, error_code.value()));
|
|
||||||
|
|
||||||
ValueType callback_arguments[1];
|
|
||||||
callback_arguments[0] = object;
|
|
||||||
|
|
||||||
Function<T>::callback(protected_ctx, protected_callback, typename T::Object(), 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::shared_ptr<SyncUser> user = sync_config->user;
|
|
||||||
if (user && user->state() != SyncUser::State::Error) {
|
|
||||||
if (auto session = user->session_for_on_disk_path(realm->config().path)) {
|
|
||||||
if (!Value::is_null(ctx, session_callback)) {
|
|
||||||
FunctionType session_callback_func = Value::to_function(ctx, session_callback);
|
|
||||||
auto syncSession = create_object<T, SessionClass<T>>(ctx, new WeakSession(session));
|
|
||||||
ValueType callback_arguments[1];
|
|
||||||
callback_arguments[0] = syncSession;
|
|
||||||
Function<T>::callback(protected_ctx, session_callback_func, typename T::Object(), 1, callback_arguments);
|
|
||||||
}
|
|
||||||
|
|
||||||
session->wait_for_download_completion(std::move(wait_handler));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ObjectType object = Object::create_empty(protected_ctx);
|
|
||||||
Object::set_property(protected_ctx, object, "message",
|
|
||||||
Value::from_string(protected_ctx, "Cannot asynchronously open synced Realm because the associated session previously experienced a fatal error"));
|
|
||||||
Object::set_property(protected_ctx, object, "errorCode", Value::from_number(protected_ctx, 1));
|
|
||||||
|
|
||||||
ValueType callback_arguments[1];
|
|
||||||
callback_arguments[0] = object;
|
|
||||||
Function<T>::callback(protected_ctx, protected_callback, protected_this, 1, callback_arguments);
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
Function<T>::callback(ctx, callback_function, this_object, 0, nullptr);
|
Protected<FunctionType> protected_callback(ctx, callback_function);
|
||||||
|
Protected<ObjectType> protected_this(ctx, this_object);
|
||||||
|
Protected<typename T::GlobalContext> protected_ctx(Context<T>::get_global_context(ctx));
|
||||||
|
|
||||||
|
std::shared_ptr<SyncUser> user = sync_config->user;
|
||||||
|
if (user && user->state() != SyncUser::State::Error) {
|
||||||
|
if (auto session = user->session_for_on_disk_path(realm->config().path)) {
|
||||||
|
if (!Value::is_null(ctx, session_callback)) {
|
||||||
|
FunctionType session_callback_func = Value::to_function(ctx, session_callback);
|
||||||
|
auto syncSession = create_object<T, SessionClass<T>>(ctx, new WeakSession(session));
|
||||||
|
ValueType callback_arguments[1];
|
||||||
|
callback_arguments[0] = syncSession;
|
||||||
|
Function<T>::callback(protected_ctx, session_callback_func, typename T::Object(), 1, callback_arguments);
|
||||||
|
}
|
||||||
|
|
||||||
|
EventLoopDispatcher<WaitHandler> wait_handler([=](std::error_code error_code) {
|
||||||
|
HANDLESCOPE
|
||||||
|
if (!error_code) {
|
||||||
|
//success
|
||||||
|
Function<T>::callback(protected_ctx, protected_callback, typename T::Object(), 0, nullptr);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
//fail
|
||||||
|
ObjectType object = Object::create_empty(protected_ctx);
|
||||||
|
Object::set_property(protected_ctx, object, "message", Value::from_string(protected_ctx, error_code.message()));
|
||||||
|
Object::set_property(protected_ctx, object, "errorCode", Value::from_number(protected_ctx, error_code.value()));
|
||||||
|
|
||||||
|
ValueType callback_arguments[1];
|
||||||
|
callback_arguments[0] = object;
|
||||||
|
|
||||||
|
Function<T>::callback(protected_ctx, protected_callback, typename T::Object(), 1, callback_arguments);
|
||||||
|
}
|
||||||
|
// Ensure that the session remains alive until the callback has had an opportunity to reopen the Realm
|
||||||
|
// with the appropriate schema.
|
||||||
|
(void)session;
|
||||||
|
});
|
||||||
|
session->wait_for_download_completion(std::move(wait_handler));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ObjectType object = Object::create_empty(protected_ctx);
|
||||||
|
Object::set_property(protected_ctx, object, "message",
|
||||||
|
Value::from_string(protected_ctx, "Cannot asynchronously open synced Realm because the associated session previously experienced a fatal error"));
|
||||||
|
Object::set_property(protected_ctx, object, "errorCode", Value::from_number(protected_ctx, 1));
|
||||||
|
|
||||||
|
ValueType callback_arguments[1];
|
||||||
|
callback_arguments[0] = object;
|
||||||
|
Function<T>::callback(protected_ctx, protected_callback, protected_this, 1, callback_arguments);
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
void RealmClass<T>::objects(ContextType ctx, ObjectType this_object, Arguments args, ReturnValue &return_value) {
|
void RealmClass<T>::objects(ContextType ctx, ObjectType this_object, Arguments args, ReturnValue &return_value) {
|
||||||
|
@ -207,6 +207,21 @@ module.exports = {
|
|||||||
TestCase.assertEqual(realm.readOnly, true);
|
TestCase.assertEqual(realm.readOnly, true);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
testRealmOpen: function() {
|
||||||
|
let realm = new Realm({schema: [schemas.TestObject], schemaVersion: 1});
|
||||||
|
realm.write(() => {
|
||||||
|
realm.create('TestObject', [1])
|
||||||
|
});
|
||||||
|
realm.close();
|
||||||
|
|
||||||
|
return Realm.open({schema: [schemas.TestObject], schemaVersion: 2}).then(realm => {
|
||||||
|
const objects = realm.objects('TestObject');
|
||||||
|
TestCase.assertEqual(objects.length, 1);
|
||||||
|
TestCase.assertEqual(objects[0].doubleCol, 1.0);
|
||||||
|
realm.close();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
testDefaultPath: function() {
|
testDefaultPath: function() {
|
||||||
const defaultPath = Realm.defaultPath;
|
const defaultPath = Realm.defaultPath;
|
||||||
let defaultRealm = new Realm({schema: []});
|
let defaultRealm = new Realm({schema: []});
|
||||||
|
@ -169,6 +169,50 @@ module.exports = {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
testRealmOpenWithExistingLocalRealm() {
|
||||||
|
if (!isNodeProccess) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const username = uuid();
|
||||||
|
const realmName = uuid();
|
||||||
|
const expectedObjectsCount = 3;
|
||||||
|
|
||||||
|
let user, config;
|
||||||
|
return runOutOfProcess(__dirname + '/download-api-helper.js', username, realmName, REALM_MODULE_PATH)
|
||||||
|
.then(() => Realm.Sync.User.login('http://localhost:9080', username, 'password'))
|
||||||
|
.then(u => {
|
||||||
|
user = u;
|
||||||
|
const accessTokenRefreshed = this;
|
||||||
|
let successCounter = 0;
|
||||||
|
|
||||||
|
config = {
|
||||||
|
sync: { user, url: `realm://localhost:9080/~/${realmName}` },
|
||||||
|
schema: [{ name: 'Dog', properties: { name: 'string' } }],
|
||||||
|
schemaVersion: 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Open the Realm with a schema version of 1, then immediately close it.
|
||||||
|
// This verifies that Realm.open doesn't hit issues when the schema version
|
||||||
|
// of an existing, local Realm is different than the one passed in the configuration.
|
||||||
|
let realm = new Realm(config);
|
||||||
|
realm.close();
|
||||||
|
|
||||||
|
config.schemaVersion = 2;
|
||||||
|
return Realm.open(config)
|
||||||
|
}).then(realm => {
|
||||||
|
let actualObjectsCount = realm.objects('Dog').length;
|
||||||
|
TestCase.assertEqual(actualObjectsCount, expectedObjectsCount, "Synced realm does not contain the expected objects count");
|
||||||
|
|
||||||
|
const session = realm.syncSession;
|
||||||
|
TestCase.assertInstanceOf(session, Realm.Sync.Session);
|
||||||
|
TestCase.assertEqual(session.user.identity, user.identity);
|
||||||
|
TestCase.assertEqual(session.config.url, config.sync.url);
|
||||||
|
TestCase.assertEqual(session.config.user.identity, config.sync.user.identity);
|
||||||
|
TestCase.assertEqual(session.state, 'active');
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
testRealmOpenAsync() {
|
testRealmOpenAsync() {
|
||||||
if (!isNodeProccess) {
|
if (!isNodeProccess) {
|
||||||
return;
|
return;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user