//////////////////////////////////////////////////////////////////////////// // // 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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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(); const credentials = Realm.Sync.Credentials.usernamePassword(username, 'password', true); return Realm.Sync.User.login('http://localhost:9080', credentials).then((user) => { assertIsUser(user); return Realm.Sync.User.login('http://localhost:9080', credentials) .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.Credentials.usernamePassword(undefined, 'password')); }, testRegisterMissingPassword() { TestCase.assertThrows(() => Realm.Sync.Credentials.usernamePassword(uuid(), undefined)); }, testRegisterServerOffline() { // Because it waits for answer this takes some time.. return Realm.Sync.User.login('http://fake_host.local', Realm.Sync.Credentials.anonymous()) .catch((e) => {}) .then((user) => { if (user) { throw new Error('should not have been able to register'); }}) }, testLogin() { const username = uuid(); const registerCredentials = Realm.Sync.Credentials.usernamePassword(username, 'password', true); // Create user, logout the new user, then login return Realm.Sync.User.login('http://localhost:9080', registerCredentials).then((user) => { user.logout(); const loginCredentials = Realm.Sync.Credentials.usernamePassword(username, 'password', false); return Realm.Sync.User.login('http://localhost:9080', loginCredentials); }).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.login('http://localhost:9080', Realm.Sync.Credentials.usernamePassword(username, 'password', true)).then((user) => { user.logout(); return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.usernamePassword(username, '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', Realm.Sync.Credentials.usernamePassword('foo', 'pass', false)) .then((user) => { throw new Error(user); }) .catch((e) => assertIsAuthError(e, 611, "The provided credentials are invalid or the user does not exist.")) }, testLoginTowardsMisbehavingServer() { // Try authenticating towards a server thats clearly not ROS return Realm.Sync.User.login('https://github.com/realm/realm-js', Realm.Sync.Credentials.anonymous()) .catch((e) => { assertIsError(e); TestCase.assertEqual( e.message, "Could not authenticate: Realm Object Server didn't respond with valid JSON" ); }); }, 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.login('http://localhost:9080', Realm.Sync.Credentials.jwt(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.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()); }).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.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).then((user) => { user1 = user; assertIsSameUser(Realm.Sync.User.current, user1); return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()); }).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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"); } const credentials = Realm.Sync.Credentials.usernamePassword(global.testAdminUserInfo.username, global.testAdminUserInfo.password); return Realm.Sync.User.login('http://localhost:9080', credentials).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"); } const credentials = Realm.Sync.Credentials.usernamePassword(global.testAdminUserInfo.username, global.testAdminUserInfo.password); return Realm.Sync.User.login('http://localhost:9080', credentials).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()).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() { return Realm.Sync.User.login('http://localhost:9080', Realm.Sync.Credentials.anonymous()) .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!' }); }); }); }); }, */ };