realm-js/tests/js/user-tests.js

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!' });
});
});
});
}, */
};