//////////////////////////////////////////////////////////////////////////// // // 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. // //////////////////////////////////////////////////////////////////////////// 'use strict'; var Realm = require('realm'); var TestCase = require('./asserts'); var schemas = require('./schemas'); let pathSeparator = '/'; if (typeof process === 'object' && process.platform === 'win32') { pathSeparator = '\\'; } module.exports = { testRealmConstructor: function() { var realm = new Realm({schema: []}); TestCase.assertTrue(realm instanceof Realm); TestCase.assertEqual(typeof Realm, 'function'); TestCase.assertTrue(Realm instanceof Function); }, testRealmConstructorPath: function() { TestCase.assertThrows(function() { new Realm(''); }, 'Realm cannot be created with an invalid path'); TestCase.assertThrows(function() { new Realm('test1.realm', 'invalidArgument'); }, 'Realm constructor can only have 0 or 1 argument(s)'); var defaultRealm = new Realm({schema: []}); TestCase.assertEqual(defaultRealm.path, Realm.defaultPath); var defaultRealm2 = new Realm(); TestCase.assertEqual(defaultRealm2.path, Realm.defaultPath); var defaultDir = Realm.defaultPath.substring(0, Realm.defaultPath.lastIndexOf(pathSeparator) + 1) var testPath = 'test1.realm'; var realm = new Realm({schema: [], path: testPath}); TestCase.assertEqual(realm.path, defaultDir + testPath); var testPath2 = 'test2.realm'; var realm2 = new Realm({schema: [], path: testPath2}); TestCase.assertEqual(realm2.path, defaultDir + testPath2); }, testRealmConstructorSchemaVersion: function() { var defaultRealm = new Realm({schema: []}); TestCase.assertEqual(defaultRealm.schemaVersion, 0); TestCase.assertThrows(function() { new Realm({schemaVersion: 1, schema: []}); }, "Realm already opened at a different schema version"); TestCase.assertEqual(new Realm().schemaVersion, 0); TestCase.assertEqual(new Realm({schemaVersion: 0}).schemaVersion, 0); var realm = new Realm({path: 'test1.realm', schema: [], schemaVersion: 1}); TestCase.assertEqual(realm.schemaVersion, 1); TestCase.assertEqual(realm.schema.length, 0); realm.close(); // FIXME - enable once realm initialization supports schema comparison // TestCase.assertThrows(function() { // realm = new Realm({path: testPath, schema: [schemas.TestObject], schemaVersion: 1}); // }, "schema changes require updating the schema version"); realm = new Realm({path: 'test1.realm', schema: [schemas.TestObject], schemaVersion: 2}); realm.write(function() { realm.create('TestObject', {doubleCol: 1}); }); TestCase.assertEqual(realm.objects('TestObject')[0].doubleCol, 1); TestCase.assertEqual(realm.schemaVersion, 2); TestCase.assertEqual(realm.schema.length, 1); }, testRealmConstructorDynamicSchema: function() { var realm = new Realm({schema: [schemas.TestObject]}); realm.write(function() { realm.create('TestObject', [1]) }); realm.close(); realm = new Realm(); var objects = realm.objects('TestObject'); TestCase.assertEqual(objects.length, 1); TestCase.assertEqual(objects[0].doubleCol, 1.0); }, testRealmConstructorSchemaValidation: function() { TestCase.assertThrows(function() { new Realm({schema: schemas.AllTypes}); }, 'The schema should be an array'); TestCase.assertThrows(function() { new Realm({schema: ['SomeType']}); }, 'The schema should be an array of objects'); TestCase.assertThrows(function() { new Realm({schema: [{}]}); }, 'The schema should be an array of ObjectSchema objects'); TestCase.assertThrows(function() { new Realm({schema: [{name: 'SomeObject'}]}); }, 'The schema should be an array of ObjectSchema objects'); TestCase.assertThrows(function() { new Realm({schema: [{properties: {intCol: 'int'}}]}); }, 'The schema should be an array of ObjectSchema objects'); // linkingObjects property where the source property is missing TestCase.assertThrows(function() { new Realm({schema: [{ name: 'InvalidObject', properties: { linkingObjects: {type:'linkingObjects', objectType: 'InvalidObject', property: 'nosuchproperty'} } }]}); }, "Property 'InvalidObject.nosuchproperty' declared as origin of linking objects property 'InvalidObject.linkingObjects' does not exist"); // linkingObjects property where the source property is not a link TestCase.assertThrows(function() { new Realm({schema: [{ name: 'InvalidObject', properties: { integer: 'int', linkingObjects: {type:'linkingObjects', objectType: 'InvalidObject', property: 'integer'} } }]}); }, "Property 'InvalidObject.integer' declared as origin of linking objects property 'InvalidObject.linkingObjects' is not a link") // linkingObjects property where the source property links elsewhere TestCase.assertThrows(function() { new Realm({schema: [{ name: 'InvalidObject', properties: { link: 'IntObject', linkingObjects: {type:'linkingObjects', objectType: 'InvalidObject', property: 'link'} } }, { name: 'IntObject', properties: { integer: 'int' } }]}); }, "Property 'InvalidObject.link' declared as origin of linking objects property 'InvalidObject.linkingObjects' links to type 'IntObject'") }, testRealmConstructorReadOnly: function() { var realm = new Realm({schema: [schemas.TestObject]}); realm.write(function() { realm.create('TestObject', [1]) }); TestCase.assertEqual(realm.readOnly, false); realm.close(); realm = new Realm({readOnly: true, schema: [schemas.TestObject]}); var objects = realm.objects('TestObject'); TestCase.assertEqual(objects.length, 1); TestCase.assertEqual(objects[0].doubleCol, 1.0); TestCase.assertEqual(realm.readOnly, true); TestCase.assertThrows(function() { realm.write(function() {}); }); realm.close(); realm = new Realm({readOnly: true}); TestCase.assertEqual(realm.schema.length, 1); TestCase.assertEqual(realm.readOnly, true); }, testDefaultPath: function() { var defaultPath = Realm.defaultPath; var defaultRealm = new Realm({schema: []}); TestCase.assertEqual(defaultRealm.path, Realm.defaultPath); try { var newPath = Realm.defaultPath.substring(0, defaultPath.lastIndexOf(pathSeparator) + 1) + 'default2.realm'; Realm.defaultPath = newPath; defaultRealm = new Realm({schema: []}); TestCase.assertEqual(defaultRealm.path, newPath, "should use updated default realm path"); TestCase.assertEqual(Realm.defaultPath, newPath, "defaultPath should have been updated"); } finally { Realm.defaultPath = defaultPath; } }, testRealmSchemaVersion: function() { TestCase.assertEqual(Realm.schemaVersion(Realm.defaultPath), -1); var realm = new Realm({schema: []}); TestCase.assertEqual(realm.schemaVersion, 0); TestCase.assertEqual(Realm.schemaVersion(Realm.defaultPath), 0); realm = new Realm({schema: [], schemaVersion: 2, path: 'another.realm'}); TestCase.assertEqual(realm.schemaVersion, 2); TestCase.assertEqual(Realm.schemaVersion('another.realm'), 2); }, testRealmWrite: function() { var realm = new Realm({schema: [schemas.IntPrimary, schemas.AllTypes, schemas.TestObject, schemas.LinkToAllTypes]}); // exceptions should be propogated TestCase.assertThrows(function() { realm.write(function() { realm.invalid(); }); }); // writes should be possible after caught exception realm.write(function() { realm.create('TestObject', {doubleCol: 1}); }); TestCase.assertEqual(1, realm.objects('TestObject').length); realm.write(function() { // nested transactions not supported TestCase.assertThrows(function() { realm.write(function() {}); }); }); }, testRealmCreate: function() { var realm = new Realm({schema: [schemas.TestObject]}); TestCase.assertThrows(function() { realm.create('TestObject', {doubleCol: 1}); }, 'can only create inside a write transaction'); realm.write(function() { realm.create('TestObject', {doubleCol: 1}); realm.create('TestObject', {doubleCol: 2}); }); var objects = realm.objects('TestObject'); TestCase.assertEqual(objects.length, 2, 'wrong object count'); TestCase.assertEqual(objects[0].doubleCol, 1, 'wrong object property value'); TestCase.assertEqual(objects[1].doubleCol, 2, 'wrong object property value'); }, testRealmCreatePrimaryKey: function() { var realm = new Realm({schema: [schemas.IntPrimary]}); realm.write(function() { var obj0 = realm.create('IntPrimaryObject', { primaryCol: 0, valueCol: 'val0', }); TestCase.assertThrows(function() { realm.create('IntPrimaryObject', { primaryCol: 0, valueCol: 'val0', }); }, 'cannot create object with conflicting primary key'); realm.create('IntPrimaryObject', { primaryCol: 1, valueCol: 'val1', }, true); var objects = realm.objects('IntPrimaryObject'); TestCase.assertEqual(objects.length, 2); realm.create('IntPrimaryObject', { primaryCol: 0, valueCol: 'newVal0', }, true); TestCase.assertEqual(obj0.valueCol, 'newVal0'); TestCase.assertEqual(objects.length, 2); realm.create('IntPrimaryObject', {primaryCol: 0}, true); TestCase.assertEqual(obj0.valueCol, 'newVal0'); }); }, testRealmCreateOptionals: function() { var realm = new Realm({schema: [schemas.NullableBasicTypes, schemas.LinkTypes, schemas.TestObject]}); var basic, links; realm.write(function() { basic = realm.create('NullableBasicTypesObject', {}); links = realm.create('LinkTypesObject', {}); }); for (var name in schemas.NullableBasicTypes.properties) { TestCase.assertEqual(basic[name], null); } TestCase.assertEqual(links.objectCol, null); TestCase.assertEqual(links.arrayCol.length, 0); }, testRealmCreateUpsert: function() { var realm = new Realm({schema: [schemas.IntPrimary, schemas.StringPrimary, schemas.AllTypes, schemas.TestObject, schemas.LinkToAllTypes]}); realm.write(function() { var values = { primaryCol: '0', boolCol: true, intCol: 1, floatCol: 1.1, doubleCol: 1.11, stringCol: '1', dateCol: new Date(1), dataCol: new ArrayBuffer(1), objectCol: {doubleCol: 1}, arrayCol: [], }; var obj0 = realm.create('AllTypesObject', values); TestCase.assertThrows(function() { realm.create('AllTypesObject', values); }, 'cannot create object with conflicting primary key'); var obj1 = realm.create('AllTypesObject', { primaryCol: '1', boolCol: false, intCol: 2, floatCol: 2.2, doubleCol: 2.22, stringCol: '2', dateCol: new Date(2), dataCol: new ArrayBuffer(2), objectCol: {doubleCol: 0}, arrayCol: [{doubleCol: 2}], }, true); var objects = realm.objects('AllTypesObject'); TestCase.assertEqual(objects.length, 2); realm.create('AllTypesObject', { primaryCol: '0', boolCol: false, intCol: 2, floatCol: 2.2, doubleCol: 2.22, stringCol: '2', dateCol: new Date(2), dataCol: new ArrayBuffer(2), objectCol: null, arrayCol: [{doubleCol: 2}], }, true); TestCase.assertEqual(objects.length, 2); TestCase.assertEqual(obj0.stringCol, '2'); TestCase.assertEqual(obj0.boolCol, false); TestCase.assertEqual(obj0.intCol, 2); TestCase.assertEqualWithTolerance(obj0.floatCol, 2.2, 0.000001); TestCase.assertEqualWithTolerance(obj0.doubleCol, 2.22, 0.000001); TestCase.assertEqual(obj0.dateCol.getTime(), 2); TestCase.assertEqual(obj0.dataCol.byteLength, 2); TestCase.assertEqual(obj0.objectCol, null); TestCase.assertEqual(obj0.arrayCol.length, 1); realm.create('AllTypesObject', {primaryCol: '0'}, true); realm.create('AllTypesObject', {primaryCol: '1'}, true); TestCase.assertEqual(obj0.stringCol, '2'); TestCase.assertEqual(obj0.objectCol, null); TestCase.assertEqual(obj1.objectCol.doubleCol, 0); realm.create('AllTypesObject', { primaryCol: '0', stringCol: '3', objectCol: {doubleCol: 0}, }, true); TestCase.assertEqual(obj0.stringCol, '3'); TestCase.assertEqual(obj0.boolCol, false); TestCase.assertEqual(obj0.intCol, 2); TestCase.assertEqualWithTolerance(obj0.floatCol, 2.2, 0.000001); TestCase.assertEqualWithTolerance(obj0.doubleCol, 2.22, 0.000001); TestCase.assertEqual(obj0.dateCol.getTime(), 2); TestCase.assertEqual(obj0.dataCol.byteLength, 2); TestCase.assertEqual(obj0.objectCol.doubleCol, 0); TestCase.assertEqual(obj0.arrayCol.length, 1); realm.create('AllTypesObject', {primaryCol: '0', objectCol: undefined}, true); realm.create('AllTypesObject', {primaryCol: '1', objectCol: null}, true); TestCase.assertEqual(obj0.objectCol, null); TestCase.assertEqual(obj1.objectCol, null); // test with string primaries var obj =realm.create('StringPrimaryObject', { primaryCol: '0', valueCol: 0 }); TestCase.assertEqual(obj.valueCol, 0); realm.create('StringPrimaryObject', { primaryCol: '0', valueCol: 1 }, true); TestCase.assertEqual(obj.valueCol, 1); }); }, testRealmWithIndexedProperties: function() { var realm = new Realm({schema: [schemas.IndexedTypes]}); realm.write(function() { realm.create('IndexedTypesObject', {boolCol: true, intCol: 1, stringCol: '1', dateCol: new Date(1)}); }); var NotIndexed = { name: 'NotIndexedObject', properties: { floatCol: {type: 'float', indexed: false} } }; new Realm({schema: [NotIndexed], path: '1.realm'}); var IndexedSchema = { name: 'IndexedSchema', }; TestCase.assertThrows(function() { IndexedSchema.properties = { floatCol: {type: 'float', indexed: true} }; new Realm({schema: [IndexedSchema], path: '2.realm'}); }); TestCase.assertThrows(function() { IndexedSchema.properties = { doubleCol: {type: 'double', indexed: true} } new Realm({schema: [IndexedSchema], path: '3.realm'}); }); TestCase.assertThrows(function() { IndexedSchema.properties = { dataCol: {type: 'data', indexed: true} } new Realm({schema: [IndexedSchema], path: '4.realm'}); }); // primary key IndexedSchema.properties = { intCol: {type: 'int', indexed: true} }; IndexedSchema.primaryKey = 'intCol'; // Test this doesn't throw new Realm({schema: [IndexedSchema], path: '5.realm'}); }, testRealmCreateWithDefaults: function() { var realm = new Realm({schema: [schemas.DefaultValues, schemas.TestObject]}); var createAndTestObject = function() { var obj = realm.create('DefaultValuesObject', {}); var properties = schemas.DefaultValues.properties; TestCase.assertEqual(obj.boolCol, properties.boolCol.default); TestCase.assertEqual(obj.intCol, properties.intCol.default); TestCase.assertEqualWithTolerance(obj.floatCol, properties.floatCol.default, 0.000001); TestCase.assertEqualWithTolerance(obj.doubleCol, properties.doubleCol.default, 0.000001); TestCase.assertEqual(obj.stringCol, properties.stringCol.default); TestCase.assertEqual(obj.dateCol.getTime(), properties.dateCol.default.getTime()); TestCase.assertEqual(obj.dataCol.byteLength, properties.dataCol.default.byteLength); TestCase.assertEqual(obj.objectCol.doubleCol, properties.objectCol.default.doubleCol); TestCase.assertEqual(obj.nullObjectCol, null); TestCase.assertEqual(obj.arrayCol.length, properties.arrayCol.default.length); TestCase.assertEqual(obj.arrayCol[0].doubleCol, properties.arrayCol.default[0].doubleCol); }; realm.write(createAndTestObject); // Defaults should still work when creating another Realm instance. realm = new Realm(); realm.write(createAndTestObject); }, testRealmCreateWithChangingDefaults: function() { var objectSchema = { name: 'IntObject', properties: { intCol: {type: 'int', default: 1}, } }; var realm = new Realm({schema: [objectSchema]}); var createAndTestObject = function() { var object = realm.create('IntObject', {}); TestCase.assertEqual(object.intCol, objectSchema.properties.intCol.default); }; realm.write(createAndTestObject); objectSchema.properties.intCol.default++; realm = new Realm({schema: [objectSchema]}); realm.write(createAndTestObject); }, testRealmCreateWithConstructor: function() { var customCreated = 0; function CustomObject() { customCreated++; } CustomObject.schema = { name: 'CustomObject', properties: { intCol: 'int' } } function InvalidObject() { return {}; } TestCase.assertThrows(function() { new Realm({schema: [InvalidObject]}); }); InvalidObject.schema = { name: 'InvalidObject', properties: { intCol: 'int' } }; var realm = new Realm({schema: [CustomObject, InvalidObject]}); realm.write(function() { var object = realm.create('CustomObject', {intCol: 1}); TestCase.assertTrue(object instanceof CustomObject); TestCase.assertTrue(Object.getPrototypeOf(object) == CustomObject.prototype); TestCase.assertEqual(customCreated, 1); // Should be able to create object by passing in constructor. object = realm.create(CustomObject, {intCol: 2}); TestCase.assertTrue(object instanceof CustomObject); TestCase.assertTrue(Object.getPrototypeOf(object) == CustomObject.prototype); TestCase.assertEqual(customCreated, 2); }); TestCase.assertThrows(function() { realm.write(function() { realm.create('InvalidObject', {intCol: 1}); }); }); // Only the original constructor should be valid. function InvalidCustomObject() {} InvalidCustomObject.schema = CustomObject.schema; TestCase.assertThrows(function() { realm.write(function() { realm.create(InvalidCustomObject, {intCol: 1}); }); }); // The constructor should still work when creating another Realm instance. realm = new Realm(); TestCase.assertTrue(realm.objects('CustomObject')[0] instanceof CustomObject); TestCase.assertTrue(realm.objects(CustomObject).length > 0); }, testRealmCreateWithChangingConstructor: function() { function CustomObject() {} CustomObject.schema = { name: 'CustomObject', properties: { intCol: 'int' } }; var realm = new Realm({schema: [CustomObject]}); realm.write(function() { var object = realm.create('CustomObject', {intCol: 1}); TestCase.assertTrue(object instanceof CustomObject); }); function NewCustomObject() {} NewCustomObject.schema = CustomObject.schema; realm = new Realm({schema: [NewCustomObject]}); realm.write(function() { var object = realm.create('CustomObject', {intCol: 1}); TestCase.assertTrue(object instanceof NewCustomObject); }); }, testRealmDelete: function() { var realm = new Realm({schema: [schemas.TestObject]}); realm.write(function() { for (var i = 0; i < 10; i++) { realm.create('TestObject', {doubleCol: i}); } }); var objects = realm.objects('TestObject'); TestCase.assertThrows(function() { realm.delete(objects[0]); }, 'can only delete in a write transaction'); realm.write(function() { TestCase.assertThrows(function() { realm.delete(); }); realm.delete(objects[0]); TestCase.assertEqual(objects.length, 9, 'wrong object count'); TestCase.assertEqual(objects[0].doubleCol, 9, "wrong property value"); TestCase.assertEqual(objects[1].doubleCol, 1, "wrong property value"); realm.delete([objects[0], objects[1]]); TestCase.assertEqual(objects.length, 7, 'wrong object count'); TestCase.assertEqual(objects[0].doubleCol, 7, "wrong property value"); TestCase.assertEqual(objects[1].doubleCol, 8, "wrong property value"); var threeObjects = realm.objects('TestObject').filtered("doubleCol < 5"); TestCase.assertEqual(threeObjects.length, 3, "wrong results count"); realm.delete(threeObjects); TestCase.assertEqual(objects.length, 4, 'wrong object count'); TestCase.assertEqual(threeObjects.length, 0, 'threeObject should have been deleted'); var o = objects[0]; realm.delete(o); TestCase.assertThrows(function() { realm.delete(o); }); }); }, testDeleteAll: function() { var realm = new Realm({schema: [schemas.TestObject, schemas.IntPrimary]}); realm.write(function() { realm.create('TestObject', {doubleCol: 1}); realm.create('TestObject', {doubleCol: 2}); realm.create('IntPrimaryObject', {primaryCol: 2, valueCol: 'value'}); }); TestCase.assertEqual(realm.objects('TestObject').length, 2); TestCase.assertEqual(realm.objects('IntPrimaryObject').length, 1); TestCase.assertThrows(function() { realm.deleteAll(); }, 'can only deleteAll in a write transaction'); realm.write(function() { realm.deleteAll(); }); TestCase.assertEqual(realm.objects('TestObject').length, 0); TestCase.assertEqual(realm.objects('IntPrimaryObject').length, 0); }, testRealmObjects: function() { var realm = new Realm({schema: [schemas.PersonObject, schemas.DefaultValues, schemas.TestObject]}); realm.write(function() { realm.create('PersonObject', {name: 'Ari', age: 10}); realm.create('PersonObject', {name: 'Tim', age: 11}); realm.create('PersonObject', {name: 'Bjarne', age: 12}); realm.create('PersonObject', {name: 'Alex', age: 12, married: true}); }); // Should be able to pass constructor for getting objects. var objects = realm.objects(schemas.PersonObject); TestCase.assertTrue(objects[0] instanceof schemas.PersonObject); function InvalidPerson() {} InvalidPerson.schema = schemas.PersonObject.schema; TestCase.assertThrows(function() { realm.objects(); }); TestCase.assertThrows(function() { realm.objects([]); }); TestCase.assertThrows(function() { realm.objects('InvalidClass'); }); TestCase.assertThrows(function() { realm.objects('PersonObject', 'truepredicate'); }); TestCase.assertThrows(function() { realm.objects(InvalidPerson); }); var person = realm.objects('PersonObject')[0]; var listenerCallback = () => {}; realm.addListener('change', listenerCallback); // The tests below assert that everthing throws when // operating on a closed realm realm.close(); TestCase.assertThrows(function() { console.log("Name: ", person.name); }); TestCase.assertThrows(function() { realm.objects('PersonObject'); }); TestCase.assertThrows(function() { realm.addListener('change', () => {}); }); TestCase.assertThrows(function() { realm.create('PersonObject', {name: 'Ari', age: 10}); }); TestCase.assertThrows(function() { realm.delete(person); }); TestCase.assertThrows(function() { realm.deleteAll(); }); TestCase.assertThrows(function() { realm.write(() => {}); }); TestCase.assertThrows(function() { realm.removeListener('change', listenerCallback); }); TestCase.assertThrows(function() { realm.removeAllListeners(); }); }, testRealmObjectForPrimaryKey: function() { var realm = new Realm({schema: [schemas.IntPrimary, schemas.StringPrimary, schemas.TestObject]}); realm.write(function() { realm.create('IntPrimaryObject', {primaryCol: 0, valueCol: 'val0'}); realm.create('IntPrimaryObject', {primaryCol: 1, valueCol: 'val1'}); realm.create('StringPrimaryObject', {primaryCol: '', valueCol: -1}); realm.create('StringPrimaryObject', {primaryCol: 'val0', valueCol: 0}); realm.create('StringPrimaryObject', {primaryCol: 'val1', valueCol: 1}); realm.create('TestObject', {doubleCol: 0}); }); TestCase.assertEqual(realm.objectForPrimaryKey('IntPrimaryObject', -1), undefined); TestCase.assertEqual(realm.objectForPrimaryKey('IntPrimaryObject', 0).valueCol, 'val0'); TestCase.assertEqual(realm.objectForPrimaryKey('IntPrimaryObject', 1).valueCol, 'val1'); TestCase.assertEqual(realm.objectForPrimaryKey('StringPrimaryObject', 'invalid'), undefined); TestCase.assertEqual(realm.objectForPrimaryKey('StringPrimaryObject', '').valueCol, -1); TestCase.assertEqual(realm.objectForPrimaryKey('StringPrimaryObject', 'val0').valueCol, 0); TestCase.assertEqual(realm.objectForPrimaryKey('StringPrimaryObject', 'val1').valueCol, 1); TestCase.assertThrows(function() { realm.objectForPrimaryKey('TestObject', 0); }); TestCase.assertThrows(function() { realm.objectForPrimaryKey(); }); TestCase.assertThrows(function() { realm.objectForPrimaryKey('IntPrimary'); }); TestCase.assertThrows(function() { realm.objectForPrimaryKey('InvalidClass', 0); }); }, testNotifications: function() { var realm = new Realm({schema: []}); var notificationCount = 0; var notificationName; realm.addListener('change', function(realm, name) { notificationCount++; notificationName = name; }); TestCase.assertEqual(notificationCount, 0); realm.write(function() {}); TestCase.assertEqual(notificationCount, 1); TestCase.assertEqual(notificationName, 'change'); var secondNotificationCount = 0; function secondNotification() { secondNotificationCount++; } // The listener should only be added once. realm.addListener('change', secondNotification); realm.addListener('change', secondNotification); realm.write(function() {}); TestCase.assertEqual(notificationCount, 2); TestCase.assertEqual(secondNotificationCount, 1); realm.removeListener('change', secondNotification); realm.write(function() {}); TestCase.assertEqual(notificationCount, 3); TestCase.assertEqual(secondNotificationCount, 1); realm.removeAllListeners(); realm.write(function() {}); TestCase.assertEqual(notificationCount, 3); TestCase.assertEqual(secondNotificationCount, 1); TestCase.assertThrows(function() { realm.addListener('invalid', function() {}); }); realm.addListener('change', function() { throw new Error('error'); }); TestCase.assertThrows(function() { realm.write(function() {}); }); }, testSchema: function() { var originalSchema = [schemas.TestObject, schemas.BasicTypes, schemas.NullableBasicTypes, schemas.IndexedTypes, schemas.IntPrimary, schemas.PersonObject, schemas.LinkTypes, schemas.LinkingObjectsObject]; var schemaMap = {}; originalSchema.forEach(function(objectSchema) { if (objectSchema.schema) { // for PersonObject schemaMap[objectSchema.schema.name] = objectSchema; } else { schemaMap[objectSchema.name] = objectSchema; } }); var realm = new Realm({schema: originalSchema}); var schema = realm.schema; TestCase.assertEqual(schema.length, originalSchema.length); function isString(val) { return typeof val === 'string' || val instanceof String; } function verifyObjectSchema(returned) { var original = schemaMap[returned.name]; if (original.schema) { original = original.schema; } TestCase.assertEqual(returned.primaryKey, original.primaryKey); for (var propName in returned.properties) { var prop1 = returned.properties[propName]; var prop2 = original.properties[propName]; if (prop1.type == 'object') { TestCase.assertEqual(prop1.objectType, isString(prop2) ? prop2 : prop2.objectType); TestCase.assertEqual(prop1.optional, true); } else if (prop1.type == 'list') { TestCase.assertEqual(prop1.objectType, prop2.objectType); TestCase.assertEqual(prop1.optional, undefined); } else if (prop1.type == 'linking objects') { TestCase.assertEqual(prop1.objectType, prop2.objectType); TestCase.assertEqual(prop1.property, prop2.property); TestCase.assertEqual(prop1.optional, undefined); } else { TestCase.assertEqual(prop1.type, isString(prop2) ? prop2 : prop2.type); TestCase.assertEqual(prop1.optional, prop2.optional || undefined); } TestCase.assertEqual(prop1.indexed, prop2.indexed || undefined); } } for (var i = 0; i < originalSchema.length; i++) { verifyObjectSchema(schema[i]); } }, testCopyBundledRealmFiles: function() { Realm.copyBundledRealmFiles(); var realm = new Realm({path: 'dates-v5.realm', schema: [schemas.DateObject]}); TestCase.assertEqual(realm.objects('Date').length, 2); TestCase.assertEqual(realm.objects('Date')[0].currentDate.getTime(), 1462500087955); var newDate = new Date(1); realm.write(function() { realm.objects('Date')[0].currentDate = newDate; }); realm.close(); // copy should not overwrite existing files Realm.copyBundledRealmFiles(); realm = new Realm({path: 'dates-v5.realm', schema: [schemas.DateObject]}); TestCase.assertEqual(realm.objects('Date')[0].currentDate.getTime(), 1); }, testErrorMessageFromInvalidWrite: function() { var realm = new Realm({schema: [schemas.PersonObject]}); TestCase.assertThrowsException(function() { realm.write(function () { var p1 = realm.create('PersonObject', { name: 'Ari', age: 10 }); p1.age = "Ten"; }); }, new Error("PersonObject.age must be of type 'number', got (Ten)")); }, testErrorMessageFromInvalidCreate: function() { var realm = new Realm({schema: [schemas.PersonObject]}); TestCase.assertThrowsException(function() { realm.write(function () { var p1 = realm.create('PersonObject', { name: 'Ari', age: 'Ten' }); }); }, new Error("PersonObject.age must be of type 'number', got (Ten)")); }, testValidTypesForListProperties: function() { var realm = new Realm({schema: [schemas.PersonObject]}); realm.write(function () { var p1 = realm.create('PersonObject', { name: 'Ari', age: 10 }); var p2 = realm.create('PersonObject', { name: 'Harold', age: 55, children: realm.objects('PersonObject').filtered('age < 15') }); TestCase.assertEqual(p2.children.length, 1); var p3 = realm.create('PersonObject', { name: 'Wendy', age: 52, children: p2.children }); TestCase.assertEqual(p3.children.length, 1); }); }, testEmpty: function() { const realm = new Realm({schema: [schemas.PersonObject]}); TestCase.assertTrue(realm.empty); realm.write(() => realm.create('PersonObject', { name: 'Ari', age: 10 })); TestCase.assertTrue(!realm.empty); realm.write(() => realm.delete(realm.objects('PersonObject'))); TestCase.assertTrue(realm.empty); }, testManualTransaction: function() { const realm = new Realm({schema: [schemas.TestObject]}); TestCase.assertTrue(realm.empty); realm.beginTransaction(); realm.create('TestObject', {doubleCol: 3.1415}); realm.commitTransaction(); TestCase.assertEqual(realm.objects('TestObject').length, 1); }, testCancelTransaction: function() { const realm = new Realm({schema: [schemas.TestObject]}); TestCase.assertTrue(realm.empty); realm.beginTransaction(); realm.create('TestObject', {doubleCol: 3.1415}); realm.cancelTransaction(); TestCase.assertTrue(realm.empty); }, testIsInTransaction: function() { const realm = new Realm({schema: [schemas.TestObject]}); TestCase.assertTrue(!realm.isInTransaction); realm.beginTransaction(); TestCase.assertTrue(realm.isInTransaction); realm.cancelTransaction(); TestCase.assertTrue(!realm.isInTransaction); }, testCompact: function() { var wasCalled = false; const count = 1000; // create compactable Realm const realm1 = new Realm({schema: [schemas.StringOnly]}); realm1.write(() => { realm1.create('StringOnlyObject', { stringCol: 'A' }); for (var i = 0; i < count; i++) { realm1.create('StringOnlyObject', { stringCol: 'ABCDEFG' }); } realm1.create('StringOnlyObject', { stringCol: 'B' }); }); realm1.close(); // open Realm and see if it is compacted var shouldCompact = function(totalBytes, usedBytes) { wasCalled = true; const fiveHundredKB = 500*1024; return (totalBytes > fiveHundredKB) && (usedBytes / totalBytes) < 0.2; }; const realm2 = new Realm({schema: [schemas.StringOnly], shouldCompactOnLaunch: shouldCompact}); TestCase.assertTrue(wasCalled); TestCase.assertEqual(realm2.objects('StringOnlyObject').length, count + 2); // we don't check if the file is smaller as we assume that Object Store does it realm2.close(); }, testManualCompact: function() { const realm1 = new Realm({schema: [schemas.StringOnly]}); realm1.write(() => { realm1.create('StringOnlyObject', { stringCol: 'A' }); }); TestCase.assertTrue(realm1.compact()); realm1.close(); const realm2 = new Realm({schema: [schemas.StringOnly]}); TestCase.assertEqual(realm2.objects('StringOnlyObject').length, 1); realm2.close(); }, testManualCompactInWrite: function() { const realm = new Realm({schema: [schemas.StringOnly]}); realm.write(() => { TestCase.assertThrows(() => { realm.compact(); }); }); TestCase.assertTrue(realm.empty); }, testManualCompactMultipleInstances: function() { const realm1 = new Realm({schema: [schemas.StringOnly]}); const realm2 = new Realm({schema: [schemas.StringOnly]}); TestCase.assertThrows(realm1.compact()); }, testRealmDeleteFileDefaultConfigPath: function() { const config = {schema: [schemas.TestObject]}; const realm = new Realm(config); realm.write(function() { realm.create('TestObject', {doubleCol: 1}); }); TestCase.assertEqual(realm.objects('TestObject').length, 1); realm.close(); Realm.deleteFile(config); const realm2 = new Realm(config); TestCase.assertEqual(realm2.objects('TestObject').length, 0); realm.close(); }, testRealmDeleteFileCustomConfigPath: function() { const config = {schema: [schemas.TestObject], path: 'test-realm-delete-file.realm'}; const realm = new Realm(config); realm.write(function() { realm.create('TestObject', {doubleCol: 1}); }); TestCase.assertEqual(realm.objects('TestObject').length, 1); realm.close(); Realm.deleteFile(config); const realm2 = new Realm(config); TestCase.assertEqual(realm2.objects('TestObject').length, 0); realm.close(); } };