SemanticScuttle/includes/js/dojo/tests/data/ItemFileWriteStore.js

1403 lines
50 KiB
JavaScript
Raw Normal View History

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;
}
]
);
}