436 lines
14 KiB
JavaScript
436 lines
14 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) {
|
|
var 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);
|
|
}
|
|
}
|
|
|
|
function failOnError(error) {
|
|
if (error) {
|
|
throw new Error(`Unexpected error in test: ${error}\n${error.stack}`);
|
|
}
|
|
}
|
|
|
|
// Test the given requestFunc, passing it the given callback after it's been wrapped appropriately.
|
|
// This function makes sure that errors thrown in the async callback rejects the promise (making tests actually run).
|
|
function callbackTest(requestFunc, callback) {
|
|
return new Promise((resolve, reject) => {
|
|
function callbackWrapper() {
|
|
try {
|
|
callback.apply(this, Array.from(arguments));
|
|
resolve();
|
|
}
|
|
catch (e) {
|
|
reject(e);
|
|
}
|
|
}
|
|
requestFunc(callbackWrapper);
|
|
});
|
|
}
|
|
|
|
module.exports = {
|
|
|
|
testLogout() {
|
|
var username = uuid();
|
|
return callbackTest((callback) => Realm.Sync.User.register('http://localhost:9080', username, 'password', callback), (error, user) => {
|
|
failOnError(error);
|
|
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(function() {
|
|
var _realm = new Realm({sync: {user: user, url: 'realm://localhost:9080/~/test'}});
|
|
});
|
|
})
|
|
},
|
|
|
|
testRegisterUser() {
|
|
var username = uuid();
|
|
return callbackTest((callback) => Realm.Sync.User.register('http://localhost:9080', username, 'password', callback), (error, user) => {
|
|
failOnError(error);
|
|
assertIsUser(user);
|
|
|
|
// Can we open a realm with the registered user?
|
|
var realm = new Realm({sync: {user: user, url: 'realm://localhost:9080/~/test'}});
|
|
TestCase.assertInstanceOf(realm, Realm);
|
|
});
|
|
},
|
|
|
|
testRegisterExistingUser() {
|
|
var username = uuid();
|
|
return new Promise((resolve, reject) => {
|
|
Realm.Sync.User.register('http://localhost:9080', username, 'password', (error, user) => {
|
|
failOnError(error);
|
|
assertIsUser(user);
|
|
|
|
Realm.Sync.User.register('http://localhost:9080', username, 'password', (error, user) => {
|
|
try {
|
|
assertIsAuthError(error, 611, "The provided credentials are invalid or the user does not exist.");
|
|
TestCase.assertUndefined(user);
|
|
resolve();
|
|
} catch(e) {
|
|
reject(e);
|
|
}
|
|
});
|
|
});
|
|
});
|
|
},
|
|
|
|
testRegisterMissingUsername() {
|
|
return new Promise((resolve, _reject) => {
|
|
TestCase.assertThrows(() => {
|
|
Realm.Sync.User.register('http://localhost:9080', undefined, 'password', () => {});
|
|
});
|
|
resolve();
|
|
});
|
|
},
|
|
|
|
testRegisterMissingPassword() {
|
|
var username = uuid();
|
|
return new Promise((resolve, _reject) => {
|
|
TestCase.assertThrows(() => {
|
|
Realm.Sync.User.register('http://localhost:9080', username, undefined, () => {});
|
|
});
|
|
resolve();
|
|
});
|
|
},
|
|
|
|
testRegisterServerOffline() {
|
|
var username = uuid();
|
|
// Because it waits for answer this takes some time..
|
|
return callbackTest((callback) => Realm.Sync.User.register('http://fake_host.local', username, 'password', callback), (error, user) => {
|
|
assertIsError(error);
|
|
TestCase.assertUndefined(user);
|
|
});
|
|
},
|
|
|
|
testLogin() {
|
|
var username = uuid();
|
|
// Create user, logout the new user, then login
|
|
return callbackTest((callback) => Realm.Sync.User.register('http://localhost:9080', username, 'password', callback), (error, user) => {
|
|
failOnError(error);
|
|
user.logout();
|
|
|
|
Realm.Sync.User.login('http://localhost:9080', username, 'password', (error, user) => {
|
|
failOnError(error);
|
|
assertIsUser(user);
|
|
|
|
// Can we open a realm with the logged-in user?
|
|
var realm = new Realm({ sync: { user: user, url: 'realm://localhost:9080/~/test' } });
|
|
TestCase.assertInstanceOf(realm, Realm);
|
|
realm.close();
|
|
});
|
|
});
|
|
},
|
|
|
|
testLoginMissingUsername() {
|
|
return new Promise((resolve, _reject) => {
|
|
TestCase.assertThrows(() => {
|
|
Realm.Sync.User.login('http://localhost:9080', undefined, 'password', () => {});
|
|
});
|
|
resolve();
|
|
});
|
|
},
|
|
|
|
testLoginMissingPassword() {
|
|
var username = uuid();
|
|
return new Promise((resolve, _reject) => {
|
|
TestCase.assertThrows(() => {
|
|
Realm.Sync.User.login('http://localhost:9080', username, undefined, () => {});
|
|
});
|
|
resolve();
|
|
});
|
|
},
|
|
|
|
testLoginNonExistingUser() {
|
|
return callbackTest((callback) => Realm.Sync.User.login('http://localhost:9080', 'does_not', 'exist', callback), (error, user) => {
|
|
assertIsAuthError(error, 611, "The provided credentials are invalid or the user does not exist.");
|
|
TestCase.assertUndefined(user);
|
|
});
|
|
},
|
|
|
|
testLoginServerOffline() {
|
|
var username = uuid();
|
|
|
|
// Because it waits for answer this takes some time..
|
|
return new Promise((resolve, reject) => {
|
|
Realm.Sync.User.register('http://fake_host.local', username, 'password', (error, user) => {
|
|
try {
|
|
assertIsError(error);
|
|
TestCase.assertUndefined(user);
|
|
resolve();
|
|
}
|
|
catch (e) { reject(e) }
|
|
});
|
|
});
|
|
},
|
|
|
|
testAll() {
|
|
return new Promise((resolve, reject) => {
|
|
let all;
|
|
all = Realm.Sync.User.all;
|
|
TestCase.assertArrayLength(Object.keys(all), 0);
|
|
|
|
callbackTest((callback) => Realm.Sync.User.register('http://localhost:9080', uuid(), 'password', callback), (error, user1) => {
|
|
failOnError(error);
|
|
|
|
all = Realm.Sync.User.all;
|
|
TestCase.assertArrayLength(Object.keys(all), 1);
|
|
assertIsSameUser(all[user1.identity], user1);
|
|
|
|
Realm.Sync.User.register('http://localhost:9080', uuid(), 'password', (error, user2) => {
|
|
failOnError(error);
|
|
|
|
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);
|
|
resolve();
|
|
});
|
|
}).catch(e => reject(e));
|
|
});
|
|
},
|
|
|
|
testCurrent() {
|
|
return new Promise((resolve, reject) => {
|
|
TestCase.assertUndefined(Realm.Sync.User.current);
|
|
|
|
callbackTest((callback) => Realm.Sync.User.register('http://localhost:9080', uuid(), 'password', callback), (error, user1) => {
|
|
failOnError(error);
|
|
assertIsSameUser(Realm.Sync.User.current, user1);
|
|
|
|
Realm.Sync.User.register('http://localhost:9080', uuid(), 'password', (error, user2) => {
|
|
failOnError(error);
|
|
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);
|
|
|
|
resolve();
|
|
|
|
});
|
|
}).catch(e => reject(e));
|
|
});
|
|
},
|
|
|
|
testManagementRealm() {
|
|
return callbackTest((callback) => Realm.Sync.User.register('http://localhost:9080', uuid(), 'password', callback), (error, user) => {
|
|
failOnError(error);
|
|
|
|
let realm = user.openManagementRealm();
|
|
TestCase.assertInstanceOf(realm, Realm);
|
|
|
|
let objectSchemaNames = realm.schema.map(o => o.name);
|
|
TestCase.assertArraysEqual(objectSchemaNames, [ 'PermissionChange', 'PermissionOffer', 'PermissionOfferResponse' ]);
|
|
});
|
|
},
|
|
|
|
testRetrieveAccount() {
|
|
return new Promise((resolve, reject) => {
|
|
if (!isNodeProcess) {
|
|
resolve();
|
|
}
|
|
|
|
if (!global.testAdminUserInfo) {
|
|
reject("Test requires an admin user");
|
|
}
|
|
|
|
Realm.Sync.User.login('http://localhost:9080', global.testAdminUserInfo.username, global.testAdminUserInfo.password, (error, user) => {
|
|
if (error) {
|
|
reject(error);
|
|
}
|
|
|
|
TestCase.assertTrue(user.isAdmin, "Test requires an admin user");
|
|
|
|
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);
|
|
resolve();
|
|
})
|
|
.catch(e => reject(e));
|
|
})
|
|
});
|
|
},
|
|
|
|
testRetrieveNotExistingAccount() {
|
|
return new Promise((resolve, reject) => {
|
|
if (!isNodeProcess) {
|
|
resolve();
|
|
}
|
|
|
|
if (!global.testAdminUserInfo) {
|
|
reject("Test requires an admin user");
|
|
}
|
|
|
|
Realm.Sync.User.login('http://localhost:9080', global.testAdminUserInfo.username, global.testAdminUserInfo.password, (error, user) => {
|
|
if (error) {
|
|
reject(error);
|
|
}
|
|
|
|
TestCase.assertTrue(user.isAdmin, "Test requires an admin user");
|
|
|
|
let notExistingUsername = uuid();
|
|
user.retrieveAccount('password', notExistingUsername)
|
|
.then(account => {
|
|
reject("Retrieving not existing account should fail");
|
|
})
|
|
.catch(e => {
|
|
try {
|
|
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");
|
|
}
|
|
catch (e) {
|
|
reject(e);
|
|
}
|
|
resolve()
|
|
});
|
|
})
|
|
});
|
|
},
|
|
|
|
/* 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!' });
|
|
});
|
|
});
|
|
});
|
|
}, */
|
|
|
|
};
|
|
|