summaryrefslogtreecommitdiffstatshomepage
path: root/includes/js/dojo/tests/data/ItemFileWriteStore.js
diff options
context:
space:
mode:
Diffstat (limited to 'includes/js/dojo/tests/data/ItemFileWriteStore.js')
-rw-r--r--includes/js/dojo/tests/data/ItemFileWriteStore.js1402
1 files changed, 1402 insertions, 0 deletions
diff --git a/includes/js/dojo/tests/data/ItemFileWriteStore.js b/includes/js/dojo/tests/data/ItemFileWriteStore.js
new file mode 100644
index 0000000..9dd938e
--- /dev/null
+++ b/includes/js/dojo/tests/data/ItemFileWriteStore.js
@@ -0,0 +1,1402 @@
+if(!dojo._hasResource["tests.data.ItemFileWriteStore"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
+dojo._hasResource["tests.data.ItemFileWriteStore"] = true;
+dojo.provide("tests.data.ItemFileWriteStore");
+dojo.require("tests.data.readOnlyItemFileTestTemplates");
+
+dojo.require("dojo.data.ItemFileWriteStore");
+dojo.require("dojo.data.api.Read");
+dojo.require("dojo.data.api.Identity");
+dojo.require("dojo.data.api.Write");
+dojo.require("dojo.data.api.Notification");
+
+
+// First, make sure ItemFileWriteStore can still pass all the same unit tests
+// that we use for its superclass, ItemFileReadStore:
+tests.data.readOnlyItemFileTestTemplates.registerTestsForDatastore("dojo.data.ItemFileWriteStore");
+
+tests.data.ItemFileWriteStore.getTestData = function(name){
+ var data = {};
+ if(name === "reference_integrity"){
+ if(dojo.isBrowser){
+ data = {url: dojo.moduleUrl("tests", "data/reference_integrity.json").toString() };
+ }else{
+ data =
+ { data: {
+ "identifier": "id",
+ "label": "name",
+ "items": [
+ {"id": 1, "name": "Item 1"},
+ {"id": 2, "name": "Item 2"},
+ {"id": 3, "name": "Item 3"},
+ {"id": 4, "name": "Item 4"},
+ {"id": 5, "name": "Item 5"},
+ {"id": 6, "name": "Item 6"},
+ {"id": 7, "name": "Item 7"},
+ {"id": 8, "name": "Item 8"},
+ {"id": 9, "name": "Item 9"},
+ {"id": 10, "name": "Item 10", "friends": [{"_reference": 1},{"_reference": 3},{"_reference": 5}]},
+ {"id": 11, "name": "Item 11", "friends": [{"_reference": 10}], "siblings": [{"_reference": 10}]},
+ {"id": 12, "name": "Item 12", "friends": [{"_reference": 3},{"_reference": 7}], "enemies": [{"_reference": 10}]},
+ {"id": 13, "name": "Item 13", "friends": [{"_reference": 10}]},
+ {"id": 14, "name": "Item 14", "friends": [{"_reference": 11}]},
+ {"id": 15, "name": "item 15", "friends": [{"id": 16, "name": "Item 16"}]}
+ ]
+ }
+ }
+ }
+ }
+ return data;
+};
+
+
+// Now run some tests that are specific to the write-access features:
+doh.register("tests.data.ItemFileWriteStore",
+ [
+ function test_getFeatures(){
+ // summary:
+ // Simple test of the getFeatures function of the store
+ // description:
+ // Simple test of the getFeatures function of the store
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var features = store.getFeatures();
+
+ // make sure we have the expected features:
+ doh.assertTrue(features["dojo.data.api.Read"] != null);
+ doh.assertTrue(features["dojo.data.api.Identity"] != null);
+ doh.assertTrue(features["dojo.data.api.Write"] != null);
+ doh.assertTrue(features["dojo.data.api.Notification"] != null);
+ doh.assertFalse(features["iggy"]);
+
+ // and only the expected features:
+ var count = 0;
+ for(var i in features){
+ doh.assertTrue((i === "dojo.data.api.Read" ||
+ i === "dojo.data.api.Identity" ||
+ i === "dojo.data.api.Write" ||
+ i === "dojo.data.api.Notification"));
+ count++;
+ }
+ doh.assertEqual(count, 4);
+ },
+ function testWriteAPI_setValue(){
+ // summary:
+ // Simple test of the setValue API
+ // description:
+ // Simple test of the setValue API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onComplete(items, request){
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "capital", "Cairo"));
+
+ // FIXME:
+ // Okay, so this seems very odd. Maybe I'm just being dense.
+ // These tests works:
+ doh.assertEqual(store.isDirty(item), false);
+ doh.assertTrue(store.isDirty(item) == false);
+ // But these seemingly equivalent tests will not work:
+ // doh.assertFalse(store.isDirty(item));
+ // doh.assertTrue(!(store.isDirty(item)));
+ //
+ // All of which seems especially weird, given that this *does* work:
+ doh.assertFalse(store.isDirty());
+
+
+ doh.assertTrue(store.isDirty(item) == false);
+ doh.assertTrue(!store.isDirty());
+ store.setValue(item, "capital", "New Cairo");
+ doh.assertTrue(store.isDirty(item));
+ doh.assertTrue(store.isDirty());
+ doh.assertEqual(store.getValue(item, "capital").toString(), "New Cairo");
+ deferred.callback(true);
+ }
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_setValues(){
+ // summary:
+ // Simple test of the setValues API
+ // description:
+ // Simple test of the setValues API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onComplete(items, request){
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Egypt"));
+ doh.assertTrue(store.isDirty(item) == false);
+ doh.assertTrue(!store.isDirty());
+ store.setValues(item, "name", ["Egypt 1", "Egypt 2"]);
+ doh.assertTrue(store.isDirty(item));
+ doh.assertTrue(store.isDirty());
+ var values = store.getValues(item, "name");
+ doh.assertTrue(values[0] == "Egypt 1");
+ doh.assertTrue(values[1] == "Egypt 2");
+ deferred.callback(true);
+ }
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_unsetAttribute(){
+ // summary:
+ // Simple test of the unsetAttribute API
+ // description:
+ // Simple test of the unsetAttribute API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onComplete(items, request) {
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Egypt"));
+ doh.assertTrue(store.isDirty(item) == false);
+ doh.assertTrue(!store.isDirty());
+ store.unsetAttribute(item, "name");
+ doh.assertTrue(store.isDirty(item));
+ doh.assertTrue(store.isDirty());
+ doh.assertTrue(!store.hasAttribute(item, "name"));
+ deferred.callback(true);
+ }
+ function onError(error, request) {
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_newItem(){
+ // summary:
+ // Simple test of the newItem API
+ // description:
+ // Simple test of the newItem API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ doh.assertTrue(!store.isDirty());
+
+ var onNewInvoked = false;
+ store.onNew = function(newItem, parentInfo){
+
+ doh.assertTrue(newItem !== null);
+ doh.assertTrue(parentInfo === null);
+ doh.assertTrue(store.isItem(newItem));
+ onNewInvoked = true;
+ };
+ var canada = store.newItem({name: "Canada", abbr:"ca", capital:"Ottawa"});
+ doh.assertTrue(onNewInvoked);
+
+ doh.assertTrue(store.isDirty(canada));
+ doh.assertTrue(store.isDirty());
+ doh.assertTrue(store.getValues(canada, "name") == "Canada");
+ function onComplete(items, request){
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Canada"));
+ deferred.callback(true);
+ }
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Canada"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_newItem_withParent(){
+ // summary:
+ // Simple test of the newItem API with a parent assignment
+ // description:
+ // Simple test of the newItem API with a parent assignment
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ doh.assertTrue(!store.isDirty());
+ function onComplete(items, request){
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Egypt"));
+
+ //Attach an onNew to validate we get expected values.
+ var onNewInvoked = false;
+ store.onNew = function(newItem, parentInfo){
+ doh.assertEqual(item, parentInfo.item);
+ doh.assertEqual("cities", parentInfo.attribute);
+ doh.assertTrue(parentInfo.oldValue === undefined);
+ doh.assertTrue(parentInfo.newValue === newItem);
+ onNewInvoked = true;
+ };
+
+ //Attach an onSet and verify onSet is NOT called in this case.
+ store.onSet = function(item, attribute, oldValue, newValue){
+ doh.assertTrue(false);
+ };
+
+ //See if we can add in a new item representing the city of Cairo.
+ //This should also call the onNew set above....
+ var newItem = store.newItem({name: "Cairo", abbr: "Cairo"}, {parent: item, attribute: "cities"});
+ doh.assertTrue(onNewInvoked);
+
+ function onCompleteNewItemShallow(items, request){
+ doh.assertEqual(0, items.length);
+ function onCompleteNewItemDeep(items, request){
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertEqual("Cairo", store.getValue(item, "name"));
+ deferred.callback(true);
+ }
+ //Do a deep search now, should find the new item of the city with name attribute Cairo.
+ store.fetch({query:{name:"Cairo"}, onComplete: onCompleteNewItemDeep, onError: onError, queryOptions: {deep:true}});
+ }
+ //Do a shallow search first, should find nothing.
+ store.fetch({query:{name:"Cairo"}, onComplete: onCompleteNewItemShallow, onError: onError});
+ }
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+
+ function testWriteAPI_newItem_multiple_withParent(){
+ // summary:
+ // Simple test of the newItem API with a parent assignment multiple times.
+ // description:
+ // Simple test of the newItem API with a parent assignment multiple times.
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+
+ doh.assertTrue(!store.isDirty());
+
+ function onComplete(items, request){
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Egypt"));
+
+ //Attach an onNew to validate we get expected values.
+ store.onNew = function(newItem, parentInfo){
+ doh.assertEqual(item, parentInfo.item);
+ doh.assertEqual("cities", parentInfo.attribute);
+
+ doh.assertTrue(parentInfo.oldValue === undefined);
+
+ doh.assertTrue(parentInfo.newValue === newItem);
+ };
+
+ //See if we can add in a new item representing the city of Cairo.
+ //This should also call the onNew set above....
+ var newItem1 = store.newItem({name: "Cairo", abbr: "Cairo"}, {parent: item, attribute: "cities"});
+
+ //Attach a new onNew to validate we get expected values.
+ store.onNew = function(newItem, parentInfo){
+ doh.assertEqual(item, parentInfo.item);
+ doh.assertEqual("cities", parentInfo.attribute);
+
+ console.log(parentInfo.oldValue);
+ doh.assertTrue(parentInfo.oldValue == newItem1);
+
+ doh.assertTrue(parentInfo.newValue[0] == newItem1);
+ doh.assertTrue(parentInfo.newValue[1] == newItem);
+ };
+ var newItem2 = store.newItem({name: "Banha", abbr: "Banha"}, {parent: item, attribute: "cities"});
+
+ //Attach a new onNew to validate we get expected values.
+ store.onNew = function(newItem, parentInfo){
+ doh.assertEqual(item, parentInfo.item);
+ doh.assertEqual("cities", parentInfo.attribute);
+
+ doh.assertTrue(parentInfo.oldValue[0] == newItem1);
+ doh.assertTrue(parentInfo.oldValue[1] == newItem2);
+
+ doh.assertTrue(parentInfo.newValue[0] == newItem1);
+ doh.assertTrue(parentInfo.newValue[1] == newItem2);
+ doh.assertTrue(parentInfo.newValue[2] == newItem);
+ };
+ var newItem3 = store.newItem({name: "Damanhur", abbr: "Damanhur"}, {parent: item, attribute: "cities"});
+ deferred.callback(true);
+ }
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+
+ function testWriteAPI_deleteItem(){
+ // summary:
+ // Simple test of the deleteItem API
+ // description:
+ // Simple test of the deleteItem API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onComplete(items, request){
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Egypt"));
+ doh.assertTrue(store.isDirty(item) == false);
+ doh.assertTrue(!store.isDirty());
+ store.deleteItem(item);
+ doh.assertTrue(store.isDirty(item));
+ doh.assertTrue(store.isDirty());
+ function onCompleteToo(itemsToo, requestToo) {
+ doh.assertEqual(0, itemsToo.length);
+ deferred.callback(true);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onCompleteToo, onError: onError});
+ }
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_isDirty(){
+ // summary:
+ // Simple test of the isDirty API
+ // description:
+ // Simple test of the isDirty API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onComplete(items, request) {
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Egypt"));
+ store.setValue(item, "name", "Egypt 2");
+ doh.assertTrue(store.getValue(item, "name") == "Egypt 2");
+ doh.assertTrue(store.isDirty(item));
+ deferred.callback(true);
+ }
+ function onError(error, request) {
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_revert(){
+ // summary:
+ // Simple test of the revert API
+ // description:
+ // Simple test of the revert API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onComplete(items, request) {
+ doh.assertEqual(1, items.length);
+ var item = items[0];
+ doh.assertTrue(store.containsValue(item, "name", "Egypt"));
+ doh.assertTrue(store.isDirty(item) == false);
+ doh.assertTrue(!store.isDirty());
+ store.setValue(item, "name", "Egypt 2");
+ doh.assertTrue(store.getValue(item, "name") == "Egypt 2");
+ doh.assertTrue(store.isDirty(item));
+ doh.assertTrue(store.isDirty());
+ store.revert();
+
+ //Fetch again to see if it reset the state.
+ function onCompleteToo(itemsToo, requestToo){
+ doh.assertEqual(1, itemsToo.length);
+ var itemToo = itemsToo[0];
+ doh.assertTrue(store.containsValue(itemToo, "name", "Egypt"));
+ deferred.callback(true);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onCompleteToo, onError: onError});
+ }
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ store.fetch({query:{name:"Egypt"}, onComplete: onComplete, onError: onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_save(){
+ // summary:
+ // Simple test of the save API
+ // description:
+ // Simple test of the save API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onItem(item){
+ store.setValue(item, "capital", "New Cairo");
+ function onComplete() {
+ deferred.callback(true);
+ }
+ store.save({onComplete:onComplete, onError:onError});
+ }
+ store.fetchItemByIdentity({identity:"eg", onItem:onItem, onError:onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_saveVerifyState(){
+ // summary:
+ // Simple test of the save API
+ // description:
+ // Simple test of the save API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onItem(item){
+ store.setValue(item, "capital", "New Cairo");
+ function onComplete() {
+ //Check internal state. Note: Users should NOT do this, this is a UT verification
+ //of internals in this case. Ref tracker: #4394
+ doh.assertTrue(!store._saveInProgress);
+ deferred.callback(true);
+ }
+ store.save({onComplete:onComplete, onError:onError});
+ }
+ store.fetchItemByIdentity({identity:"eg", onItem:onItem, onError:onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_saveEverything(){
+ // summary:
+ // Simple test of the save API
+ // description:
+ // Simple test of the save API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+ var egypt;
+ store._saveEverything = function(saveCompleteCallback, saveFailedCallback, newFileContentString){
+ var struct = dojo.fromJson(newFileContentString);
+ doh.assertEqual(struct.identifier, store.getIdentityAttributes(egypt)[0]);
+ doh.assertEqual(struct.label, store.getLabelAttributes(egypt)[0]);
+ doh.assertEqual(struct.items.length, 7);
+
+ var cloneStore = new dojo.data.ItemFileWriteStore({data:struct});
+ function onItemClone(itemClone){
+ var egyptClone = itemClone;
+ doh.assertEqual(store.getIdentityAttributes(egypt)[0], cloneStore.getIdentityAttributes(egyptClone)[0]);
+ doh.assertEqual(store.getLabelAttributes(egypt)[0], cloneStore.getLabelAttributes(egyptClone)[0]);
+ doh.assertEqual(store.getValue(egypt, "name"), cloneStore.getValue(egyptClone, "name"));
+ }
+ cloneStore.fetchItemByIdentity({identity:"eg", onItem:onItemClone, onError:onError});
+
+ saveCompleteCallback();
+ };
+
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onItem(item){
+ egypt = item;
+ function onComplete() {
+ deferred.callback(true);
+ }
+ store.setValue(egypt, "capital", "New Cairo");
+ store.save({onComplete:onComplete, onError:onError});
+ }
+ store.fetchItemByIdentity({identity:"eg", onItem:onItem, onError:onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_saveEverything_withDateType(){
+ // summary:
+ // Simple test of the save API with a non-atomic type (Date) that has a type mapping.
+ // description:
+ // Simple test of the save API with a non-atomic type (Date) that has a type mapping.
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+ store._saveEverything = function(saveCompleteCallback, saveFailedCallback, newFileContentString){
+
+ //Now load the new data into a datastore and validate that it stored the date right.
+ var dataset = dojo.fromJson(newFileContentString);
+ var newStore = new dojo.data.ItemFileWriteStore({data: dataset});
+
+ function gotItem(item){
+ var independenceDate = newStore.getValue(item,"independence");
+ doh.assertTrue(independenceDate instanceof Date);
+ doh.assertTrue(dojo.date.compare(new Date(1993,04,24), independenceDate, "date") === 0);
+ saveCompleteCallback();
+ }
+ function failed(error, request){
+ deferred.errback(error);
+ saveFailedCallback();
+ }
+ newStore.fetchItemByIdentity({identity:"eg", onItem:gotItem, onError:failed});
+ };
+
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onItem(item){
+ function onComplete() {
+ deferred.callback(true);
+ }
+ store.setValue(item, "independence", new Date(1993,04,24));
+ store.save({onComplete:onComplete, onError:onError});
+ }
+ store.fetchItemByIdentity({identity:"eg", onItem:onItem, onError:onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_saveEverything_withCustomColorTypeSimple(){
+ // summary:
+ // Simple test of the save API with a non-atomic type (dojo.Color) that has a type mapping.
+ // description:
+ // Simple test of the save API with a non-atomic type (dojo.Color) that has a type mapping.
+
+ //Set up the store basics: What data it has, and what to do when save is called for saveEverything
+ //And how to map the 'Color' type in and out of the format.
+ //(Test of saving all to a some location...)
+ var dataset = {
+ identifier:'name',
+ items: [
+ { name:'Kermit', species:'frog', color:{_type:'Color', _value:'green'} },
+ { name:'Beaker', hairColor:{_type:'Color', _value:'red'} }
+ ]
+ };
+
+ var customTypeMap = {'Color': dojo.Color };
+
+ var store = new dojo.data.ItemFileWriteStore({
+ data:dataset,
+ typeMap: customTypeMap
+ });
+
+ store._saveEverything = function(saveCompleteCallback, saveFailedCallback, newFileContentString){
+ //Now load the new data into a datastore and validate that it stored the Color right.
+ var dataset = dojo.fromJson(newFileContentString);
+ var newStore = new dojo.data.ItemFileWriteStore({data: dataset, typeMap: customTypeMap});
+
+ function gotItem(item){
+ var hairColor = newStore.getValue(item,"hairColor");
+ doh.assertTrue(hairColor instanceof dojo.Color);
+ doh.assertEqual("rgba(255, 255, 0, 1)", hairColor.toString());
+ saveCompleteCallback();
+ }
+ function failed(error, request){
+ deferred.errback(error);
+ saveFailedCallback();
+ }
+ newStore.fetchItemByIdentity({identity:"Animal", onItem:gotItem, onError:failed});
+ };
+
+ //Add a new item with a color type, then save it.
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onComplete() {
+ deferred.callback(true);
+ }
+
+ var animal = store.newItem({name: "Animal", hairColor: new dojo.Color("yellow")});
+ store.save({onComplete:onComplete, onError:onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_saveEverything_withCustomColorTypeGeneral(){
+ // summary:
+ // Simple test of the save API with a non-atomic type (dojo.Color) that has a type mapping.
+ // description:
+ // Simple test of the save API with a non-atomic type (dojo.Color) that has a type mapping.
+
+ //Set up the store basics: What data it has, and what to do when save is called for saveEverything
+ //And how to map the 'Color' type in and out of the format.
+ //(Test of saving all to a some location...)
+ var dataset = {
+ identifier:'name',
+ items: [
+ { name:'Kermit', species:'frog', color:{_type:'Color', _value:'green'} },
+ { name:'Beaker', hairColor:{_type:'Color', _value:'red'} }
+ ]
+ };
+
+ var customTypeMap = {'Color': {
+ type: dojo.Color,
+ deserialize: function(value){
+ return new dojo.Color(value);
+ },
+ serialize: function(obj){
+ return obj.toString();
+ }
+ }
+ }
+ var store = new dojo.data.ItemFileWriteStore({
+ data:dataset,
+ typeMap: customTypeMap
+ });
+ store._saveEverything = function(saveCompleteCallback, saveFailedCallback, newFileContentString){
+ //Now load the new data into a datastore and validate that it stored the Color right.
+ var dataset = dojo.fromJson(newFileContentString);
+ var newStore = new dojo.data.ItemFileWriteStore({data: dataset, typeMap: customTypeMap});
+
+ function gotItem(item){
+ var hairColor = newStore.getValue(item,"hairColor");
+ doh.assertTrue(hairColor instanceof dojo.Color);
+ doh.assertEqual("rgba(255, 255, 0, 1)", hairColor.toString());
+ saveCompleteCallback();
+ }
+ function failed(error, request){
+ deferred.errback(error);
+ saveFailedCallback();
+ }
+ newStore.fetchItemByIdentity({identity:"Animal", onItem:gotItem, onError:failed});
+ };
+
+ //Add a new item with a color type, then save it.
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onComplete() {
+ deferred.callback(true);
+ }
+
+ var animal = store.newItem({name: "Animal", hairColor: new dojo.Color("yellow")});
+ store.save({onComplete:onComplete, onError:onError});
+ return deferred; //Object
+ },
+ function testWriteAPI_newItem_revert(){
+ // summary:
+ // Test for bug #5357. Ensure that the revert properly nulls the identity position
+ // for a new item after revert.
+ var args = {data: {
+ label:"name",
+ items:[
+ {name:'Ecuador', capital:'Quito'},
+ {name:'Egypt', capital:'Cairo'},
+ {name:'El Salvador', capital:'San Salvador'},
+ {name:'Equatorial Guinea', capital:'Malabo'},
+ {name:'Eritrea', capital:'Asmara'},
+ {name:'Estonia', capital:'Tallinn'},
+ {name:'Ethiopia', capital:'Addis Ababa'}
+ ]
+ } };
+ var store = new dojo.data.ItemFileWriteStore(args);
+
+ var newCountry = store.newItem({name: "Utopia", capitol: "Perfect"});
+
+ //DO NOT ACCESS THIS WAY. THESE ARE INTERNAL VARIABLES. DOING THIS FOR TEST PURPOSES.
+ var itemEntryNum = newCountry[store._itemNumPropName];
+ doh.assertTrue(store._arrayOfAllItems[itemEntryNum] === newCountry);
+ store.revert();
+ doh.assertTrue(store._arrayOfAllItems[itemEntryNum] === null);
+ },
+ function testNotificationAPI_onSet(){
+ // summary:
+ // Simple test of the onSet API
+ // description:
+ // Simple test of the onSet API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onItem(fetchedItem){
+ var egypt = fetchedItem;
+ var connectHandle = null;
+ function setValueHandler(item, attribute, oldValue, newValue){
+ doh.assertTrue(store.isItem(item));
+ doh.assertTrue(item == egypt);
+ doh.assertTrue(attribute == "capital");
+ doh.assertTrue(oldValue == "Cairo");
+ doh.assertTrue(newValue == "New Cairo");
+ deferred.callback(true);
+ dojo.disconnect(connectHandle);
+ }
+ connectHandle = dojo.connect(store, "onSet", setValueHandler);
+ store.setValue(egypt, "capital", "New Cairo");
+ }
+ store.fetchItemByIdentity({identity:"eg", onItem:onItem, onError:onError});
+ },
+ function testNotificationAPI_onNew(){
+ // summary:
+ // Simple test of the onNew API
+ // description:
+ // Simple test of the onNew API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ var connectHandle = null;
+ function newItemHandler(item){
+ doh.assertTrue(store.isItem(item));
+ doh.assertTrue(store.getValue(item, "name") == "Canada");
+ deferred.callback(true);
+ dojo.disconnect(connectHandle);
+ }
+ connectHandle = dojo.connect(store, "onNew", newItemHandler);
+ var canada = store.newItem({name:"Canada", abbr:"ca", capital:"Ottawa"});
+ },
+ function testNotificationAPI_onDelete(){
+ // summary:
+ // Simple test of the onDelete API
+ // description:
+ // Simple test of the onDelete API
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+
+ var deferred = new doh.Deferred();
+ function onError(error){
+ deferred.errback(error);
+ }
+ function onItem(fetchedItem){
+ var egypt = fetchedItem;
+ var connectHandle = null;
+ function deleteItemHandler(item){
+ doh.assertTrue(store.isItem(item) == false);
+ doh.assertTrue(item == egypt);
+ deferred.callback(true);
+ dojo.disconnect(connectHandle);
+ }
+ connectHandle = dojo.connect(store, "onDelete", deleteItemHandler);
+ store.deleteItem(egypt);
+ }
+ store.fetchItemByIdentity({identity:"eg", onItem:onItem, onError:onError});
+ },
+ function testReadAPI_functionConformanceToo(){
+ // summary:
+ // Simple test read API conformance. Checks to see all declared functions are actual functions on the instances.
+ // description:
+ // Simple test read API conformance. Checks to see all declared functions are actual functions on the instances.
+ var testStore = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+ var readApi = new dojo.data.api.Read();
+ var passed = true;
+
+ for(var functionName in readApi){
+ var member = readApi[functionName];
+ //Check that all the 'Read' defined functions exist on the test store.
+ if(typeof member === "function"){
+ var testStoreMember = testStore[functionName];
+ if(!(typeof testStoreMember === "function")){
+ passed = false;
+ break;
+ }
+ }
+ }
+ doh.assertTrue(passed);
+ },
+ function testWriteAPI_functionConformance(){
+ // summary:
+ // Simple test write API conformance. Checks to see all declared functions are actual functions on the instances.
+ // description:
+ // Simple test write API conformance. Checks to see all declared functions are actual functions on the instances.
+ var testStore = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+ var writeApi = new dojo.data.api.Write();
+ var passed = true;
+
+ for(var functionName in writeApi){
+ var member = writeApi[functionName];
+ //Check that all the 'Write' defined functions exist on the test store.
+ if(typeof member === "function"){
+ var testStoreMember = testStore[functionName];
+ if(!(typeof testStoreMember === "function")){
+ passed = false;
+ break;
+ }
+ }
+ }
+ doh.assertTrue(passed);
+ },
+ function testNotificationAPI_functionConformance(){
+ // summary:
+ // Simple test Notification API conformance. Checks to see all declared functions are actual functions on the instances.
+ // description:
+ // Simple test Notification API conformance. Checks to see all declared functions are actual functions on the instances.
+ var testStore = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries"));
+ var api = new dojo.data.api.Notification();
+ var passed = true;
+
+ for(var functionName in api){
+ var member = api[functionName];
+ //Check that all the 'Write' defined functions exist on the test store.
+ if(typeof member === "function"){
+ var testStoreMember = testStore[functionName];
+ if(!(typeof testStoreMember === "function")){
+ passed = false;
+ break;
+ }
+ }
+ }
+ doh.assertTrue(passed);
+ },
+ function testIdentityAPI_noIdentifierSpecified(){
+ // summary:
+ // Test for bug #3873. Given a datafile that does not specify an
+ // identifier, make sure ItemFileWriteStore auto-creates identities
+ // that are unique even after calls to deleteItem() and newItem()
+ var args = {data: {
+ label:"name",
+ items:[
+ {name:'Ecuador', capital:'Quito'},
+ {name:'Egypt', capital:'Cairo'},
+ {name:'El Salvador', capital:'San Salvador'},
+ {name:'Equatorial Guinea', capital:'Malabo'},
+ {name:'Eritrea', capital:'Asmara'},
+ {name:'Estonia', capital:'Tallinn'},
+ {name:'Ethiopia', capital:'Addis Ababa'}
+ ]
+ } };
+ var store = new dojo.data.ItemFileWriteStore(args);
+ var deferred = new doh.Deferred();
+
+ var onError = function(error, request){
+ deferred.errback(error);
+ }
+ var onComplete = function(items, request){
+ doh.assertEqual(7, items.length);
+
+ var lastItem = items[(items.length - 1)];
+ var idOfLastItem = store.getIdentity(lastItem);
+ store.deleteItem(lastItem);
+ store.newItem({name:'Canada', capital:'Ottawa'});
+
+ var onCompleteAgain = function(itemsAgain, requestAgain){
+ doh.assertEqual(7, itemsAgain.length);
+ var identitiesInUse = {};
+ for(var i = 0; i < itemsAgain.length; ++i){
+ var item = itemsAgain[i];
+ var id = store.getIdentity(item);
+ if(identitiesInUse.hasOwnProperty(id)){
+ // there should not already be an entry for this id
+ doh.assertTrue(false);
+ }else{
+ // we want to add the entry now
+ identitiesInUse[id] = item;
+ }
+ }
+ deferred.callback(true);
+ }
+ store.fetch({onComplete:onCompleteAgain, onError:onError});
+ }
+
+ store.fetch({onComplete:onComplete, onError:onError});
+ return deferred;
+ },
+ function testIdentityAPI_noIdentifierSpecified_revert(){
+ // summary:
+ // Test for bug #4691 Given a datafile that does not specify an
+ // identifier, make sure ItemFileWriteStore auto-creates identities
+ // that are unique even after calls to deleteItem() and newItem()
+ var args = {data: {
+ label:"name",
+ items:[
+ {name:'Ecuador', capital:'Quito'},
+ {name:'Egypt', capital:'Cairo'},
+ {name:'El Salvador', capital:'San Salvador'},
+ {name:'Equatorial Guinea', capital:'Malabo'},
+ {name:'Eritrea', capital:'Asmara'},
+ {name:'Estonia', capital:'Tallinn'},
+ {name:'Ethiopia', capital:'Addis Ababa'}
+ ]
+ } };
+ var store = new dojo.data.ItemFileWriteStore(args);
+ var deferred = new doh.Deferred();
+
+ var onError = function(error, request){
+ deferred.errback(error);
+ }
+ var onComplete = function(items, request){
+ doh.assertEqual(7, items.length);
+
+ var lastItem = items[(items.length - 1)];
+ var idOfLastItem = store.getIdentity(lastItem);
+ store.deleteItem(lastItem);
+ store.newItem({name:'Canada', capital:'Ottawa'});
+
+ var onCompleteAgain = function(itemsAgain, requestAgain){
+ doh.assertEqual(7, itemsAgain.length);
+ var identitiesInUse = {};
+ for(var i = 0; i < itemsAgain.length; ++i){
+ var item = itemsAgain[i];
+ var id = store.getIdentity(item);
+ if(identitiesInUse.hasOwnProperty(id)){
+ // there should not already be an entry for this id
+ doh.assertTrue(false);
+ }else{
+ // we want to add the entry now
+ identitiesInUse[id] = item;
+ }
+ }
+ //Last test, revert everything and check item sizes.
+ store.revert();
+
+ //Now call fetch again and verify store state.
+ var revertComplete = function(itemsReverted, request){
+ doh.assertEqual(7, itemsReverted.length);
+ deferred.callback(true);
+ }
+ store.fetch({onComplete:revertComplete, onError:onError});
+ }
+ store.fetch({onComplete:onCompleteAgain, onError:onError});
+ }
+ store.fetch({onComplete:onComplete, onError:onError});
+ return deferred;
+ },
+ function testReferenceIntegrity_checkReferences(){
+ // summary:
+ // Simple test to verify the references were properly resolved.
+ // description:
+ // Simple test to verify the references were properly resolved.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ function onComplete(items, request){
+
+ var item10 = null;
+ var item1 = null;
+ var item3 = null;
+ var item5 = null;
+
+ for (var i = 0; i < items.length; i++) {
+ var ident = store.getIdentity(items[i]);
+ if (ident === 10) {
+ item10 = items[i];
+ }else if (ident === 1) {
+ item1 = items[i];
+ }else if (ident === 3) {
+ item3 = items[i];
+ }else if (ident === 5) {
+ item5 = items[i];
+ }
+ }
+ var friends = store.getValues(item10, "friends");
+ doh.assertTrue(friends !== null);
+ doh.assertTrue(friends !== undefined);
+
+ doh.assertTrue(store.isItem(item10));
+ doh.assertTrue(store.isItem(item1));
+ doh.assertTrue(store.isItem(item3));
+ doh.assertTrue(store.isItem(item5));
+ var found = 0;
+ try{
+ for (var i = 0; i < friends.length; i++) {
+ if (i === 0) {
+ doh.assertTrue(store.isItem(friends[i]));
+ doh.assertEqual(friends[i], item1);
+ doh.assertEqual(store.getIdentity(friends[i]), 1);
+ found++;
+ }else if (i === 1) {
+ doh.assertTrue(store.isItem(friends[i]));
+ doh.assertEqual(friends[i], item3);
+ doh.assertEqual(store.getIdentity(friends[i]), 3);
+ found++;
+ }else if (i === 2) {
+ doh.assertTrue(store.isItem(friends[i]));
+ doh.assertEqual(friends[i], item5);
+ doh.assertEqual(store.getIdentity(friends[i]), 5);
+ found++;
+ }
+ }
+ }catch(e){
+ doh.errback(e);
+ }
+ doh.assertEqual(3, found);
+ deferred.callback(true);
+ }
+ store.fetch({onError: onError, onComplete: onComplete});
+ return deferred;
+ },
+ function testReferenceIntegrity_deleteReferencedItem(){
+ // summary:
+ // Simple test to verify the references were properly deleted.
+ // description:
+ // Simple test to verify the references were properly deleted.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ function onItem(item, request){
+ try{
+ console.log("Before delete map state is: " + dojo.toJson(item[store._reverseRefMap]));
+ store.deleteItem(item);
+ console.log("After delete map state is: " + dojo.toJson(item[store._reverseRefMap]));
+ function verifyRefDelete(items, request){
+ var passed = true;
+ for(var i = 0; i < items.length; i++){
+ var curItem = items[i];
+ var attributes = store.getAttributes(curItem);
+ for(var j = 0; j < attributes.length; j++){
+ var values = store.getValues(curItem, attributes[j]);
+ var badRef = false;
+ for(var k = 0; k < values.length; k++){
+ var value = values[k];
+ try{
+ var id = store.getIdentity(value);
+ if(id == 10){
+ badRef = true;
+ break;
+ }
+ }catch(e){/*Not an item, even a dead one, just eat it.*/}
+ }
+ if(badRef){
+ deferred.errback(new Error("Found a reference remaining to a deleted item. Failure."));
+ passed = false;
+ break;
+ }
+ }
+ }
+ if(passed){
+ deferred.callback(true);
+ }
+ }
+ store.fetch({onComplete: verifyRefDelete, onError: onError});
+ }catch(error){
+ deferred.errback(error);
+ }
+ }
+ store.fetchItemByIdentity({identity: 10, onError: onError, onItem: onItem});
+ return deferred;
+ },
+ function testReferenceIntegrity_deleteReferencedItemThenRevert(){
+ // summary:
+ // Simple test to verify the references were properly deleted.
+ // description:
+ // Simple test to verify the references were properly deleted.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ function onItem(item, request){
+ try{
+ //DO NOT EVER ACCESS THESE VARIABLES LIKE THIS!
+ //THIS IS FOR TESTING INTERNAL STATE!
+ console.log("Map before delete:");
+ store._dumpReferenceMap();
+ var beforeDelete = dojo.toJson(item[store._reverseRefMap]);
+ store.deleteItem(item);
+ console.log("Map after delete:");
+ store._dumpReferenceMap();
+ var afterDelete = dojo.toJson(item[store._reverseRefMap]);
+ store.revert();
+ console.log("Map after revert:");
+ store._dumpReferenceMap();
+ var afterRevert = dojo.toJson(item[store._reverseRefMap]);
+ doh.assertTrue(afterRevert === beforeDelete);
+ }catch(e){
+ deferred.errback(e);
+ passed = false;
+ }
+ if(passed){
+ deferred.callback(true);
+ }
+ }
+ store.fetchItemByIdentity({identity: 10, onError: onError, onItem: onItem});
+ return deferred;
+ },
+ function testReferenceIntegrity_deleteMultipleItemsWithReferencesAndRevert(){
+ // summary:
+ // Simple test to verify that a flow of deleting items with references and reverting does not damage the internal structure.
+ // Created for tracker bug: #5743
+ // description:
+ // Simple test to verify that a flow of deleting items with references and reverting does not damage the internal structure.
+ // Created for tracker bug: #5743
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.readOnlyItemFileTestTemplates.getTestData("countries_references"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ doh.assertTrue(false);
+ }
+ function onItem(item, request){
+ //Save off the located item, then locate another one (peer to Egypt)
+ doh.assertTrue(store.isItem(item));
+ var egypt = item;
+
+ function onItem2(item, request){
+ doh.assertTrue(store.isItem(item));
+ var nairobi = item;
+
+ //Delete them
+ store.deleteItem(egypt);
+ store.deleteItem(nairobi);
+ try{
+ //Revert, then do a fetch. If the internals have been damaged, this will generally
+ //cause onError to fire instead of onComplete.
+ store.revert();
+ function onComplete(items, request){
+ deferred.callback(true);
+ }
+ store.fetch({query: {name: "*"}, start: 0, count: 20, onComplete: onComplete, onError: onError});
+ }catch(e){
+ deferred.errback(e)
+ }
+ }
+ store.fetchItemByIdentity({identity: "Nairobi", onError: onError, onItem: onItem2});
+ }
+ store.fetchItemByIdentity({identity: "Egypt", onError: onError, onItem: onItem});
+ return deferred;
+ },
+ function testReferenceIntegrity_removeReferenceFromAttribute(){
+ // summary:
+ // Simple test to verify the reference removal updates the internal map.
+ // description:
+ // Simple test to verify the reference removal updates the internal map.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ doh.assertTrue(false);
+ }
+ function onItem(item, request){
+ try{
+ store.setValues(item, "friends", [null]);
+
+ function onItem2(item10, request){
+ //DO NOT EVER ACCESS THESE VARIABLES LIKE THIS!
+ //THIS IS FOR TESTING INTERNAL STATE!
+ var refMap = item10[store._reverseRefMap];
+ store._dumpReferenceMap();
+
+ console.log("MAP for Item 10 is: " + dojo.toJson(refMap));
+
+ //Assert there is no reference to item 10 in item 11's attribute 'friends'.
+ doh.assertTrue(!refMap["11"]["friends"]);
+ store.setValues(item, "siblings", [0, 1, 2]);
+ //Assert there are no more references to 10 in 11. Ergo, "11" should be a 'undefined' attribute for the map of items referencing '10'..
+ doh.assertTrue(!refMap["11"]);
+ deferred.callback(true);
+ }
+ store.fetchItemByIdentity({identity: 10, onError: onError, onItem: onItem2});
+
+ }catch(e){
+ console.debug(e);
+ deferred.errback(e);
+ doh.assertTrue(false);
+ }
+ }
+ store.fetchItemByIdentity({identity: 11, onError: onError, onItem: onItem});
+ return deferred;
+ },
+ function testReferenceIntegrity_deleteReferencedItemNonParent(){
+ // summary:
+ // Simple test to verify the references to a non-parent item was properly deleted.
+ // description:
+ // Simple test to verify the references to a non-parent item was properly deleted.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ }
+ function onItem(item, request){
+ try{
+ console.log("Reference state for item 16 is: " + dojo.toJson(item[store._reverseRefMap]));
+ store.deleteItem(item);
+ function verifyRefDelete(items, request){
+ var passed = true;
+ for(var i = 0; i < items.length; i++){
+ var curItem = items[i];
+ var attributes = store.getAttributes(curItem);
+ for(var j = 0; j < attributes.length; j++){
+ var values = store.getValues(curItem, attributes[j]);
+ var badRef = false;
+ for(var k = 0; k < values.length; k++){
+ var value = values[k];
+ try{
+ var id = store.getIdentity(value);
+ if(id == 16){
+ badRef = true;
+ break;
+ }
+ }catch(e){/*Not an item, even a dead one, just eat it.*/}
+ }
+ if(badRef){
+ deferred.errback(new Error("Found a reference remaining to a deleted item. Failure."));
+ passed = false;
+ break;
+ }
+ }
+ }
+ if(passed){
+ deferred.callback(true);
+ }
+ }
+ store.fetch({onComplete: verifyRefDelete, onError: onError});
+ }catch(error){
+ deferred.errback(error);
+ }
+ }
+ store.fetchItemByIdentity({identity: 16, onError: onError, onItem: onItem});
+ return deferred;
+ },
+ function testReferenceIntegrity_addReferenceToAttribute(){
+ // summary:
+ // Simple test to verify the reference additions can happen.
+ // description:
+ // Simple test to verify the reference additions can happen.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ doh.assertTrue(false);
+ }
+ function onComplete(items, request){
+
+ doh.assertTrue(items.length > 2);
+
+ var item1 = items[0];
+ var item2 = items[1];
+
+ //DO NOT EVER ACCESS THESE VARIABLES LIKE THIS!
+ //THIS IS FOR TESTING INTERNAL STATE!
+ console.log("Map state for Item 1 is: " + dojo.toJson(item1[store._reverseRefMap]));
+ console.log("Map state for Item 2 is: " + dojo.toJson(item2[store._reverseRefMap]));
+
+ store.setValue(item1, "siblings", item2);
+
+ //Emit the current map state for inspection.
+ console.log("Map state for Item 1 is: " + dojo.toJson(item1[store._reverseRefMap]));
+ console.log("Map state for Item 2 is: " + dojo.toJson(item2[store._reverseRefMap]));
+
+ doh.assertTrue(item2[store._reverseRefMap] != null);
+
+ //Assert there is a recorded reference to item 2 in item 1's attribute 'sibling'.
+ doh.assertTrue(item2[store._reverseRefMap][store.getIdentity(item1)]["siblings"]);
+
+ deferred.callback(true);
+ }
+ store.fetch({onError: onError, onComplete: onComplete});
+ return deferred;
+ },
+ function testReferenceIntegrity_newItemWithParentReference(){
+ // summary:
+ // Simple test to verify that newItems with a parent properly record the parent's reference in the map.
+ // description:
+ // Simple test to verify that newItems with a parent properly record the parent's reference in the map.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ doh.assertTrue(false);
+ }
+ function onItem(item, request){
+ try{
+ //Create a new item and set its parent to item 10's uncle attribute.
+ var newItem = store.newItem({id: 17, name: "Item 17"}, {parent: item, attribute: "uncles"});
+
+ //DO NOT EVER ACCESS THESE VARIABLES LIKE THIS!
+ //THIS IS FOR TESTING INTERNAL STATE!
+ //Look up the references to 17, as item 10 has one now on attribute 'uncles'
+ var refs = newItem[store._reverseRefMap];
+
+ //Assert there is a reference from 10 to item 17, on attribute uncle
+ doh.assertTrue(refs["10"]["uncles"]);
+
+ console.log("State of map of item 17 after newItem: " + dojo.toJson(refs));
+ }catch(e){
+ console.debug(e);
+ deferred.errback(e);
+ doh.assertTrue(false);
+ passed = false;
+ }
+ if(passed){
+ deferred.callback(true);
+ }
+ }
+ store.fetchItemByIdentity({identity: 10, onError: onError, onItem: onItem});
+ return deferred;
+ },
+ function testReferenceIntegrity_newItemWithReferenceToExistingItem(){
+ // summary:
+ // Simple test to verify that a new item with references to existing items properly record the references in the map.
+ // description:
+ // Simple test to verify that a new item with references to existing items properly record the references in the map.
+
+ var store = new dojo.data.ItemFileWriteStore(tests.data.ItemFileWriteStore.getTestData("reference_integrity"));
+
+ var deferred = new doh.Deferred();
+ var passed = true;
+ function onError(error, request){
+ deferred.errback(error);
+ doh.assertTrue(false);
+ }
+ function onItem(item, request){
+ try{
+ //DO NOT EVER ACCESS THESE VARIABLES LIKE THIS!
+ //THIS IS FOR TESTING INTERNAL STATE!
+ console.log("State of reference map to item 10 before newItem: " + dojo.toJson(item[store._reverseRefMap]));
+
+ //Create a new item and set its parent to item 10's uncle attribute.
+ var newItem = store.newItem({id: 17, name: "Item 17", friends: [item]});
+
+ //DO NOT EVER ACCESS THESE VARIABLES LIKE THIS!
+ //THIS IS FOR TESTING INTERNAL STATE!
+ //Look up the references to 10, as item 17 has one on friends now.
+ var refs = item[store._reverseRefMap];
+
+ //Assert there is a reference from 15 to item 10, on attribute friends
+ doh.assertTrue(refs["17"]["friends"]);
+
+ console.log("State of reference map to item 10 after newItem: " + dojo.toJson(refs));
+ }catch(e){
+ console.debug(e);
+ deferred.errback(e);
+ doh.assertTrue(false);
+ passed = false;
+ }
+ if(passed){
+ deferred.callback(true);
+ }
+ }
+ store.fetchItemByIdentity({identity: 10, onError: onError, onItem: onItem});
+ return deferred;
+ },
+ function testReferenceIntegrity_disableReferenceIntegrity(){
+ // summary:
+ // Simple test to verify reference integrity can be disabled.
+ // description:
+ // Simple test to verify reference integrity can be disabled.
+
+ var params = tests.data.ItemFileWriteStore.getTestData("reference_integrity");
+ params.referenceIntegrity = false;
+ var store = new dojo.data.ItemFileWriteStore(params);
+
+ var deferred = new doh.Deferred();
+ function onError(error, request){
+ deferred.errback(error);
+ doh.assertTrue(false);
+ }
+ function onItem(item, request){
+ //DO NOT EVER ACCESS THESE VARIABLES LIKE THIS!
+ //THIS IS FOR TESTING INTERNAL STATE!
+ if(item[store._reverseRefMap] === undefined){
+ deferred.callback(true);
+ }else{
+ deferred.errback(new Error("Disabling of reference integrity failed."));
+ }
+ }
+ store.fetchItemByIdentity({identity: 10, onError: onError, onItem: onItem});
+ return deferred;
+ }
+ ]
+);
+
+
+
+}