516 lines
19 KiB
JavaScript
516 lines
19 KiB
JavaScript
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// 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.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
/* eslint-env es6, node */
|
|
|
|
'use strict';
|
|
|
|
const Realm = require('realm');
|
|
const TestCase = require('./asserts');
|
|
const isNodeProcess = typeof process === 'object' && process + '' === '[object process]';
|
|
|
|
function uuid() {
|
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
|
|
const r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
|
|
return v.toString(16);
|
|
});
|
|
}
|
|
|
|
function assertIsUser(user, isAdmin) {
|
|
TestCase.assertType(user, 'object');
|
|
TestCase.assertType(user.token, 'string');
|
|
TestCase.assertType(user.identity, 'string');
|
|
TestCase.assertInstanceOf(user, Realm.Sync.User);
|
|
if (isAdmin !== undefined) {
|
|
TestCase.assertEqual(user.isAdmin, isAdmin);
|
|
}
|
|
}
|
|
|
|
function assertIsSameUser(value, user) {
|
|
assertIsUser(value);
|
|
TestCase.assertEqual(value.token, user.token);
|
|
TestCase.assertEqual(value.identity, user.identity);
|
|
TestCase.assertEqual(value.isAdmin, user.isAdmin);
|
|
}
|
|
|
|
function assertIsError(error, message) {
|
|
TestCase.assertInstanceOf(error, Error, 'The API should return an Error');
|
|
if (message) {
|
|
TestCase.assertEqual(error.message, message);
|
|
}
|
|
}
|
|
|
|
function assertIsAuthError(error, code, title) {
|
|
TestCase.assertInstanceOf(error, Realm.Sync.AuthError, 'The API should return an AuthError');
|
|
if (code) {
|
|
TestCase.assertEqual(error.code, code);
|
|
}
|
|
if (title) {
|
|
TestCase.assertEqual(error.title, title);
|
|
}
|
|
}
|
|
|
|
module.exports = {
|
|
|
|
testLogout() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
assertIsUser(user);
|
|
|
|
assertIsSameUser(user, Realm.Sync.User.current);
|
|
user.logout();
|
|
|
|
// Is now logged out.
|
|
TestCase.assertUndefined(Realm.Sync.User.current);
|
|
|
|
// Can we open a realm with the registered user?
|
|
TestCase.assertThrows(() => new Realm({sync: {user: user, url: 'realm://localhost:9080/~/test'}}));
|
|
});
|
|
},
|
|
|
|
testRegisterUser() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
// Can we open a realm with the registered user?
|
|
const realm = new Realm({sync: {user: user, url: 'realm://localhost:9080/~/test'}});
|
|
TestCase.assertInstanceOf(realm, Realm);
|
|
});
|
|
},
|
|
|
|
testRegisterExistingUser() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
assertIsUser(user);
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password')
|
|
.then((user) => { throw new Error(user); })
|
|
.catch((e) => {
|
|
assertIsAuthError(e, 611, "The provided credentials are invalid or the user does not exist.");
|
|
})
|
|
});
|
|
},
|
|
|
|
testRegisterMissingUsername() {
|
|
TestCase.assertThrows(() => Realm.Sync.User.register('http://localhost:9080', undefined, 'password'));
|
|
},
|
|
|
|
testRegisterMissingPassword() {
|
|
const username = uuid();
|
|
TestCase.assertThrows(() => Realm.Sync.User.register('http://localhost:9080', username, undefined));
|
|
},
|
|
|
|
testRegisterServerOffline() {
|
|
const username = uuid();
|
|
// Because it waits for answer this takes some time..
|
|
return Realm.Sync.User.register('http://fake_host.local', username, 'password')
|
|
.catch((e) => {})
|
|
.then((user) => { if (user) { throw new Error('should not have been able to register'); }})
|
|
},
|
|
|
|
testLogin() {
|
|
const username = uuid();
|
|
// Create user, logout the new user, then login
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
user.logout();
|
|
return Realm.Sync.User.login('http://localhost:9080', username, 'password');
|
|
}).then((user => {
|
|
assertIsUser(user);
|
|
// Can we open a realm with the logged-in user?
|
|
const config = user.createConfiguration({ sync: { url: 'realm://localhost:9080/~/test' }});
|
|
const realm = new Realm(config);
|
|
TestCase.assertInstanceOf(realm, Realm);
|
|
realm.close();
|
|
}))
|
|
},
|
|
|
|
testAuthenticateWithPassword() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
user.logout();
|
|
return Realm.Sync.User.authenticate('http://localhost:9080', 'password', { username: username, password: 'password' });
|
|
}).then((user => {
|
|
assertIsUser(user);
|
|
const realm = new Realm(user.createConfiguration({ sync: { url: 'realm://localhost:9080/~/test' } }));
|
|
TestCase.assertInstanceOf(realm, Realm);
|
|
realm.close();
|
|
}))
|
|
},
|
|
|
|
testLoginMissingUsername() {
|
|
TestCase.assertThrows(() => Realm.Sync.User.login('http://localhost:9080', undefined, 'password'));
|
|
},
|
|
|
|
testLoginMissingPassword() {
|
|
const username = uuid();
|
|
TestCase.assertThrows(() => Realm.Sync.User.login('http://localhost:9080', username, undefined));
|
|
},
|
|
|
|
testLoginNonExistingUser() {
|
|
return Realm.Sync.User.login('http://localhost:9080', 'does_not', 'exist')
|
|
.then((user) => { throw new Error(user); })
|
|
.catch((e) => assertIsAuthError(e, 611, "The provided credentials are invalid or the user does not exist."))
|
|
},
|
|
|
|
testLoginServerOffline() {
|
|
const username = uuid();
|
|
|
|
// Because it waits for answer this takes some time..
|
|
return Realm.Sync.User.register('http://fake_host.local', username, 'password')
|
|
.then((user) => { throw new Error(user); })
|
|
.catch((e) => assertIsError(e));
|
|
},
|
|
|
|
testLoginTowardsMisbehavingServer() {
|
|
const username = uuid();
|
|
|
|
// Try authenticating towards a server thats clearly not ROS
|
|
return Realm.Sync.User.register('https://github.com/realm/realm-js', username, 'user')
|
|
.catch((e) => {
|
|
assertIsError(e);
|
|
TestCase.assertEqual(
|
|
e.message,
|
|
"Could not authenticate: Realm Object Server didn't respond with valid JSON"
|
|
);
|
|
});
|
|
},
|
|
|
|
testAuthenticateInvalidProvider() {
|
|
return Realm.Sync.User.authenticate('http://localhost:9080', 'FooBar', {})
|
|
.then((user) => { Promise.reject() } )
|
|
.catch((e) => { Promise.resolve() } )
|
|
},
|
|
|
|
testAuthenticateJWT() {
|
|
let token = 'eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VySWQiOiJhdXN0aW5femhlbmciLCJpc0FkbWluIjp0cnVlLCJpYXQiOjE1MTI2OTI3NDl9.klca-3wYLe5mGdVk7N7dE9YRIlB1el1Dv6BxZNAKMsJ3Ms4vBTweu4-65kVJftiMrYhmSGY6QtTzqQ-xlLH4XzPd3jYIXlPQ45lxO7PW7EkJNs9m83VdcsJmHRHQ3PRP8V_mx0f2Ks4ga3xZ9IycAQB4q5NXLei_HJk8tRRJccZ6qB5nnAoD48Qu8JOEfhO596Mdoi-QCbH51iJZjgXo4gSRZ4KKK8jU0S6twLj_lf9jehENTqHDdtsRHdyCnICcPcz4AjFrNHEvUrsPkGxXSZ2BCGgDcvsSTVgGNV7rWU4IjH4FaDssenumi50R1QcZh8kiO35s9H6MngQsEm-zApRgd0V9_L3A6Ys47_crmKbunYRsATfMNBn2fKm5tS6RXvM2RN2G_Y9AkGgh2boY42CRy7HOcHby2vQ8IoQ-fZfE5xn_YYktNlKeNiCv3_-i86lANFbmB3tcdScrbjsgO6Tfg3u71VmJ_ZW1_vyMi5vCTEysLXfHG-OA85c3o8-25vcfuX5gIpbU-nMLgPagyn5w7Uazd27uhFfwepP9OMc8jz2JTlQICInLCUdESu8aG5d1F_IPUA5NU_ryPmebqUmyaRVDS8cGChxp0gZDNSiIvaggw8N2JCDGvk-s_PSG2pFGq0f4veYyWGBTHD_iX4a0UrhB471QZplRpMwvu7o'
|
|
return Realm.Sync.User.authenticate('http://localhost:9080', 'jwt', { token: token })
|
|
.then((user) => {
|
|
TestCase.assertEqual(user.identity, 'austin_zheng')
|
|
Promise.resolve()
|
|
})
|
|
.catch((e) => { Promise.reject(e) } )
|
|
},
|
|
|
|
testAll() {
|
|
const all = Realm.Sync.User.all;
|
|
TestCase.assertArrayLength(Object.keys(all), 0);
|
|
|
|
let user1;
|
|
return Realm.Sync.User.register('http://localhost:9080', uuid(), 'password').then((user) => {
|
|
const all = Realm.Sync.User.all;
|
|
TestCase.assertArrayLength(Object.keys(all), 1);
|
|
assertIsSameUser(all[user.identity], user);
|
|
user1 = user;
|
|
|
|
return Realm.Sync.User.register('http://localhost:9080', uuid(), 'password');
|
|
}).then((user2) => {
|
|
let all = Realm.Sync.User.all;
|
|
TestCase.assertArrayLength(Object.keys(all), 2);
|
|
// NOTE: the list of users is in latest-first order.
|
|
assertIsSameUser(all[user2.identity], user2);
|
|
assertIsSameUser(all[user1.identity], user1);
|
|
|
|
user2.logout();
|
|
all = Realm.Sync.User.all;
|
|
TestCase.assertArrayLength(Object.keys(all), 1);
|
|
assertIsSameUser(all[user1.identity], user1);
|
|
|
|
user1.logout();
|
|
all = Realm.Sync.User.all;
|
|
TestCase.assertArrayLength(Object.keys(all), 0);
|
|
});
|
|
},
|
|
|
|
testCurrent() {
|
|
TestCase.assertUndefined(Realm.Sync.User.current);
|
|
|
|
let user1;
|
|
return Realm.Sync.User.register('http://localhost:9080', uuid(), 'password').then((user) => {
|
|
user1 = user;
|
|
assertIsSameUser(Realm.Sync.User.current, user1);
|
|
|
|
return Realm.Sync.User.register('http://localhost:9080', uuid(), 'password');
|
|
}).then((user2) => {
|
|
TestCase.assertThrows(() => Realm.Sync.User.current, 'We expect Realm.Sync.User.current to throw if > 1 user.');
|
|
user2.logout();
|
|
|
|
assertIsSameUser(Realm.Sync.User.current, user1);
|
|
|
|
user1.logout();
|
|
TestCase.assertUndefined(Realm.Sync.User.current);
|
|
});
|
|
},
|
|
|
|
testGetExistingUser() {
|
|
let userid = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', userid, 'password').then((user) => {
|
|
let identity = user.identity;
|
|
let user1 = Realm.Sync.User._getExistingUser('http://localhost:9080', identity);
|
|
assertIsSameUser(user1, user);
|
|
user.logout();
|
|
let user2 = Realm.Sync.User._getExistingUser('http://localhost:9080', identity);
|
|
TestCase.assertUndefined(user2);
|
|
});
|
|
},
|
|
|
|
testManagementRealm() {
|
|
return Realm.Sync.User.register('http://localhost:9080', uuid(), 'password').then((user) => {
|
|
let realm = user.openManagementRealm();
|
|
TestCase.assertInstanceOf(realm, Realm);
|
|
|
|
TestCase.assertArraysEqual(realm.schema.map(o => o.name),
|
|
['PermissionChange', 'PermissionOffer', 'PermissionOfferResponse']);
|
|
});
|
|
},
|
|
|
|
testRetrieveAccount() {
|
|
if (!isNodeProcess) {
|
|
return;
|
|
}
|
|
|
|
if (!global.testAdminUserInfo) {
|
|
throw new Error("Test requires an admin user");
|
|
}
|
|
|
|
return Realm.Sync.User.login('http://localhost:9080', global.testAdminUserInfo.username, global.testAdminUserInfo.password).then((user) => {
|
|
TestCase.assertTrue(user.isAdmin, "Test requires an admin user");
|
|
|
|
return user.retrieveAccount('password', global.testAdminUserInfo.username)
|
|
}).then((account) => {
|
|
TestCase.assertEqual(account.accounts[0].provider_id, global.testAdminUserInfo.username);
|
|
TestCase.assertEqual(account.accounts[0].provider, 'password');
|
|
TestCase.assertTrue(account.is_admin);
|
|
TestCase.assertTrue(account.user_id);
|
|
});
|
|
},
|
|
|
|
testRetrieveNotExistingAccount() {
|
|
if (!isNodeProcess) {
|
|
return;
|
|
}
|
|
|
|
if (!global.testAdminUserInfo) {
|
|
throw new Error("Test requires an admin user");
|
|
}
|
|
|
|
return Realm.Sync.User.login('http://localhost:9080', global.testAdminUserInfo.username, global.testAdminUserInfo.password).then((user) => {
|
|
TestCase.assertTrue(user.isAdmin, "Test requires an admin user");
|
|
|
|
let notExistingUsername = uuid();
|
|
return user.retrieveAccount('password', notExistingUsername)
|
|
}).catch(e => {
|
|
TestCase.assertEqual(e.status, 404);
|
|
TestCase.assertEqual(e.code, 612);
|
|
TestCase.assertEqual(e.message, "The account does not exist.");
|
|
TestCase.assertEqual(e.type, "https://realm.io/docs/object-server/problems/unknown-account");
|
|
}).then(account => { if (account) { throw new Error("Retrieving nonexistent account should fail"); }});
|
|
},
|
|
|
|
testCreateConfiguration_defaultConfig() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
let config = user.createConfiguration();
|
|
TestCase.assertEqual(config.sync.url, "realm://localhost:9080/default");
|
|
TestCase.assertUndefined(config.sync.partial);
|
|
TestCase.assertFalse(config.sync.fullSynchronization);
|
|
});
|
|
},
|
|
|
|
testCreateConfiguration_useOldConfiguration() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
let config = user.createConfiguration({ sync: { url: 'http://localhost:9080/other_realm', partial: true }});
|
|
TestCase.assertEqual(config.sync.url, 'http://localhost:9080/other_realm');
|
|
TestCase.assertUndefined(config.sync.fullSynchronization);
|
|
TestCase.assertTrue(config.sync.partial);
|
|
});
|
|
},
|
|
|
|
testCreateConfiguration_settingPartialAndFullSynchronizationThrows() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
TestCase.assertThrowsContaining(() => {
|
|
let config = {
|
|
sync: {
|
|
url: 'http://localhost:9080/~/default',
|
|
partial: true,
|
|
fullSynchronization: false
|
|
}
|
|
};
|
|
user.createConfiguration(config);
|
|
}, "'partial' and 'fullSynchronization' were both set. 'partial' has been deprecated, use only 'fullSynchronization'");
|
|
});
|
|
},
|
|
|
|
testOpen_partialAndFullSynchronizationSetThrows() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
TestCase.assertThrowsContaining(() => {
|
|
new Realm({
|
|
sync: {
|
|
user: user,
|
|
url: 'http://localhost:9080/~/default',
|
|
partial: false,
|
|
fullSynchronization: true
|
|
}
|
|
})
|
|
}, "'partial' and 'fullSynchronization' were both set. 'partial' has been deprecated, use only 'fullSynchronization'");
|
|
});
|
|
},
|
|
|
|
testSerialize() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password').then((user) => {
|
|
const serialized = user.serialize();
|
|
TestCase.assertFalse(serialized.isAdmin);
|
|
TestCase.assertEqual(serialized.identity, user.identity);
|
|
TestCase.assertEqual(serialized.server, 'http://localhost:9080');
|
|
TestCase.assertEqual(serialized.refreshToken, user.token);
|
|
});
|
|
},
|
|
|
|
testDeserialize() {
|
|
const username = uuid();
|
|
return Realm.Sync.User.register('http://localhost:9080', username, 'password')
|
|
.then((user) => {
|
|
const userConfig = user.createConfiguration({
|
|
schema: [{ name: 'Dog', properties: { name: 'string' } }],
|
|
sync: {
|
|
url: 'realm://localhost:9080/~/foo',
|
|
fullSynchronization: true,
|
|
}
|
|
});
|
|
|
|
const realm = new Realm(userConfig);
|
|
realm.write(() => {
|
|
realm.create('Dog', {
|
|
name: 'Doggo'
|
|
});
|
|
});
|
|
|
|
const session = realm.syncSession;
|
|
return new Promise((resolve, reject) => {
|
|
let callback = (transferred, total) => {
|
|
if (transferred >= total) {
|
|
session.removeProgressNotification(callback);
|
|
realm.close();
|
|
Realm.deleteFile(userConfig);
|
|
resolve(user.serialize());
|
|
}
|
|
}
|
|
session.addProgressNotification('upload', 'forCurrentlyOutstandingWork', callback);
|
|
});
|
|
}).then((serialized) => {
|
|
const deserialized = Realm.Sync.User.deserialize(serialized);
|
|
const config = deserialized.createConfiguration({
|
|
schema: [{ name: 'Dog', properties: { name: 'string' } }],
|
|
sync: {
|
|
url: 'realm://localhost:9080/~/foo',
|
|
fullSynchronization: true,
|
|
}
|
|
});
|
|
|
|
return Realm.open(config);
|
|
}).then((realm) => {
|
|
const dogs = realm.objects('Dog');
|
|
TestCase.assertEqual(dogs.length, 1);
|
|
TestCase.assertEqual(dogs[0].name, 'Doggo');
|
|
});
|
|
},
|
|
|
|
testDeserializeInvalidInput() {
|
|
const dummy = {
|
|
server: '123',
|
|
identity: '123',
|
|
refreshToken: '123',
|
|
isAdmin: false,
|
|
};
|
|
|
|
for (const name of Object.getOwnPropertyNames(dummy)) {
|
|
const clone = Object.assign({}, dummy);
|
|
// Set to invalid type
|
|
clone[name] = 123;
|
|
|
|
TestCase.assertThrowsContaining(() => Realm.Sync.User.deserialize(clone), `${name} must be of type '${typeof dummy[name]}'`);
|
|
|
|
// Set to undefined
|
|
clone[name] = undefined;
|
|
TestCase.assertThrowsContaining(() => Realm.Sync.User.deserialize(clone), `${name} is required, but a value was not provided.`);
|
|
}
|
|
}
|
|
|
|
/* This test fails because of realm-object-store #243 . We should use 2 users.
|
|
testSynchronizeChangesWithTwoClientsAndOneUser() {
|
|
// Test Schema
|
|
class Foo {}
|
|
Foo.schema = {
|
|
name: 'Foo',
|
|
properties: {
|
|
string: 'string',
|
|
bars: { type: 'list', objectType: 'Bar' },
|
|
},
|
|
};
|
|
|
|
class Bar {}
|
|
Bar.schema = {
|
|
name: 'Bar',
|
|
properties: { integer: 'int' },
|
|
};
|
|
|
|
const schema = [Foo.schema, Bar.schema];
|
|
|
|
// Create a user, open two clients at different local paths, synchronize changes
|
|
const username = uuid();
|
|
return new Promise((resolve) => {
|
|
Realm.Sync.User.register('http://localhost:9080', username, 'password', (error ,user) => {
|
|
failOnError(error);
|
|
|
|
const clientA = new Realm({
|
|
path: 'testSynchronizeChangesWithTwoClientsAndOneUser_clientA.realm',
|
|
schema: schema,
|
|
sync: {
|
|
user: user,
|
|
url: 'http://localhost:9080/~/test',
|
|
},
|
|
});
|
|
|
|
const clientB = new Realm({
|
|
path: 'testSynchronizeChangesWithTwoClientsAndOneUser_clientB.realm',
|
|
schema: schema,
|
|
sync: {
|
|
user: user,
|
|
url: 'http://localhost:9080/~/test',
|
|
},
|
|
});
|
|
|
|
clientB.addListener('change', () => {
|
|
const foos = clientB.objects('Foo');
|
|
if (foos.length > 0) {
|
|
TestCase.assertEqual(foos.length, 1);
|
|
TestCase.assertEqual(foos[0].string, 'Hello, World!');
|
|
resolve();
|
|
}
|
|
});
|
|
|
|
clientA.write(() => {
|
|
clientA.create('Foo', { string: 'Hello, World!' });
|
|
});
|
|
});
|
|
});
|
|
}, */
|
|
|
|
};
|