diff options
Diffstat (limited to 'test')
| -rw-r--r-- | test/BrowserSpecs.js | 208 | ||||
| -rw-r--r-- | test/angular-mocks.js | 20 | ||||
| -rw-r--r-- | test/servicesSpec.js | 99 |
3 files changed, 323 insertions, 4 deletions
diff --git a/test/BrowserSpecs.js b/test/BrowserSpecs.js index 47cabf0f..f80de417 100644 --- a/test/BrowserSpecs.js +++ b/test/BrowserSpecs.js @@ -27,6 +27,25 @@ describe('browser', function(){ browser.startUrlWatcher(); }); + it('should contain cookie cruncher', function() { + expect(browser.cookies).toBeDefined(); + }); + + it('should be able to start cookie watcher', function() { + browser.delay = 1; + expectAsserts(2); + browser.watchCookies(function(cookies){ + assertEquals({'foo':'bar'}, cookies); + }); + browser.setTimeout = function(fn, delay){ + assertEquals(1, delay); + document.cookie = 'foo=bar'; + browser.setTimeout = function(fn, delay) {}; + fn(); + }; + browser.startCookieWatcher(); + }); + describe('outstading requests', function(){ it('should process callbacks immedietly with no outstanding requests', function(){ var callback = jasmine.createSpy('callback'); @@ -67,4 +86,193 @@ describe('browser', function(){ }); }); + + describe('cookies', function() { + + function deleteAllCookies() { + var cookies = document.cookie.split(";"); + + for (var i = 0; i < cookies.length; i++) { + var cookie = cookies[i]; + var eqPos = cookie.indexOf("="); + var name = eqPos > -1 ? cookie.substr(0, eqPos) : cookie; + document.cookie = name + "=;expires=Thu, 01 Jan 1970 00:00:00 GMT"; + } + }; + + var browser; + + beforeEach(function() { + deleteAllCookies(); + browser = new Browser({}, jqLite(document)); + expect(document.cookie).toEqual(''); + }); + + + afterEach(function() { + deleteAllCookies(); + expect(document.cookie).toEqual(''); + }); + + + describe('remove all via (null)', function() { + + it('should do nothing when no cookies are set', function() { + browser.cookies(null); + expect(document.cookie).toEqual(''); + expect(browser.cookies()).toEqual({}); + }); + + }); + + describe('remove via (cookieName, undefined)', function() { + + it('should remove a cookie when it is present', function() { + document.cookie = 'foo=bar'; + + browser.cookies('foo', undefined); + + expect(document.cookie).toEqual(''); + expect(browser.cookies()).toEqual({}); + }); + + + it('should do nothing when an nonexisting cookie is being removed', function() { + browser.cookies('doesntexist', undefined); + expect(document.cookie).toEqual(''); + expect(browser.cookies()).toEqual({}); + }); + }); + + + describe('put via (cookieName, string)', function() { + + it('should create and store a cookie', function() { + browser.cookies('cookieName', 'cookieValue'); + expect(document.cookie).toEqual('cookieName=cookieValue'); + expect(browser.cookies()).toEqual({'cookieName':'cookieValue'}); + }); + + + it('should overwrite an existing unsynced cookie', function() { + document.cookie = "cookie=new"; + + var oldVal = browser.cookies('cookie', 'newer'); + + expect(document.cookie).toEqual('cookie=newer'); + expect(browser.cookies()).toEqual({'cookie':'newer'}); + expect(oldVal).not.toBeDefined(); + }); + + it('should escape both name and value', function() { + browser.cookies('cookie1=', 'val;ue'); + browser.cookies('cookie2=bar;baz', 'val=ue'); + + var rawCookies = document.cookie.split("; "); //order is not guaranteed, so we need to parse + expect(rawCookies.length).toEqual(2); + expect(rawCookies).toContain('cookie1%3D=val%3Bue'); + expect(rawCookies).toContain('cookie2%3Dbar%3Bbaz=val%3Due'); + }); + }); + + + describe('get via (cookieName)', function() { + + it('should return undefined for nonexistent cookie', function() { + expect(browser.cookies('nonexistent')).not.toBeDefined(); + }); + + + it ('should return a value for an existing cookie', function() { + document.cookie = "foo=bar"; + browser.cookies(true); + expect(browser.cookies().foo).toEqual('bar'); + }); + + + it ('should unescape cookie values that were escaped by puts', function() { + document.cookie = "cookie2%3Dbar%3Bbaz=val%3Due"; + browser.cookies(true); + expect(browser.cookies()['cookie2=bar;baz']).toEqual('val=ue'); + }); + + + it('should preserve leading & trailing spaces in names and values', function() { + browser.cookies(' cookie name ', ' cookie value '); + expect(browser.cookies()[' cookie name ']).toEqual(' cookie value '); + expect(browser.cookies()['cookie name']).not.toBeDefined(); + }); + }); + + + describe('getAll', function() { + + it('should return cookies as hash', function() { + document.cookie = "foo1=bar1"; + document.cookie = "foo2=bar2"; + expect(browser.cookies()).toEqual({'foo1':'bar1', 'foo2':'bar2'}); + }); + + + it('should return empty hash if no cookies exist', function() { + expect(browser.cookies()).toEqual({}); + }); + }); + + + describe('watch', function() { + + it('should allow listeners to be registered', function() { + expectAsserts(1); + + browser.watchCookies(function(cookies) { + assertEquals({'aaa':'bbb'}, cookies); + }); + + browser.cookies('aaa','bbb'); + browser.cookies(); + }); + + + it('should fire listeners when cookie changes are discovered', function() { + expectAsserts(1); + + browser.watchCookies(function(cookies) { + assertEquals({'foo':'bar'}, cookies); + }); + + document.cookie = 'foo=bar'; + browser.cookies(); + }); + + + it('should not fire listeners when no cookies were changed', function() { + expectAsserts(0); + + browser.cookies(function(cookies) { + assertEquals({'shouldnt':'fire'}, cookies); + }); + + browser.cookies(true); + }); + }); + + + it('should pick up external changes made to browser cookies', function() { + browser.cookies('oatmealCookie', 'drool'); + expect(browser.cookies()).toEqual({'oatmealCookie':'drool'}); + + document.cookie = 'oatmealCookie=changed'; + browser.cookies(true); + expect(browser.cookies().oatmealCookie).toEqual('changed'); + }); + + + it('should initialize cookie cache with existing cookies', function() { + document.cookie = "existingCookie=existingValue"; + expect(browser.cookies()).toEqual({'existingCookie':'existingValue'}); + }); + + }); }); + diff --git a/test/angular-mocks.js b/test/angular-mocks.js index bac2e800..1e547f77 100644 --- a/test/angular-mocks.js +++ b/test/angular-mocks.js @@ -26,6 +26,7 @@ function MockBrowser() { var self = this, expectations = {}, requests = []; + this.isMock = true; self.url = "http://server"; self.watches = []; @@ -72,6 +73,8 @@ function MockBrowser() { requests.pop()(); } }; + + self.cookieHash = {}; } MockBrowser.prototype = { @@ -90,11 +93,28 @@ MockBrowser.prototype = { this.watches.push(fn); }, + watchCookies: function(fn) { + this.watches.push(fn); + }, + fireUrlWatchers: function() { for(var i=0; i<this.watches.length; i++) { this.watches[i](this.url); } + }, + + cookies: function(name, value) { + if (name) { + if (value == undefined) { + delete this.cookieHash[name]; + } else { + this.cookieHash[name] = ""+value; + } + } else { + return copy(this.cookieHash); + } } + }; angular.service('$browser', function(){ diff --git a/test/servicesSpec.js b/test/servicesSpec.js index 062ba8af..e1534e94 100644 --- a/test/servicesSpec.js +++ b/test/servicesSpec.js @@ -38,7 +38,7 @@ describe("service", function(){ function warn(){ logger+= 'warn;'; }; function info(){ logger+= 'info;'; }; function error(){ logger+= 'error;'; }; - var scope = createScope(null, angularService, {$window: {console:{log:log, warn:warn, info:info, error:error}}, $document:[{}]}); + var scope = createScope(null, angularService, {$window: {console:{log:log, warn:warn, info:info, error:error}}, $document:[{cookie:''}]}); scope.$log.log(); scope.$log.warn(); scope.$log.info(); @@ -49,7 +49,7 @@ describe("service", function(){ it('should use console.log if other not present', function(){ var logger = ""; function log(){ logger+= 'log;'; }; - var scope = createScope(null, angularService, {$window: {console:{log:log}}, $document:[{}]}); + var scope = createScope(null, angularService, {$window: {console:{log:log}}, $document:[{cookie:''}]}); scope.$log.log(); scope.$log.warn(); scope.$log.info(); @@ -58,7 +58,7 @@ describe("service", function(){ }); it('should use noop if no console', function(){ - var scope = createScope(null, angularService, {$window: {}, $document:[{}]}); + var scope = createScope(null, angularService, {$window: {}, $document:[{cookie:''}]}); scope.$log.log(); scope.$log.warn(); scope.$log.info(); @@ -371,4 +371,95 @@ describe("service", function(){ }); -}); + describe('$cookies', function() { + + it('should provide access to existing cookies via object properties', function(){ + expect(scope.$cookies).toEqual({}); + + scope.$browser.cookies('brandNew', 'cookie'); + //TODO: This is a hacky way of calling the watch function, once pooling is refactored, this will go away. + scope.$browser.watches[1](scope.$browser.cookies()); + + expect(scope.$cookies).toEqual({'brandNew':'cookie'}); + }); + + + it('should create or update a cookie when a value is assigned to a property', function() { + scope.$cookies.oatmealCookie = 'nom nom'; + scope.$eval(); + + expect(scope.$browser.cookies()).toEqual({'oatmealCookie':'nom nom'}); + + scope.$cookies.oatmealCookie = 'gone'; + scope.$eval(); + + expect(scope.$browser.cookies()).toEqual({'oatmealCookie':'gone'}); + }); + + + it('should turn non-string into String when creating a cookie', function() { + scope.$cookies.nonString = [1, 2, 3]; + scope.$eval(); + expect(scope.$browser.cookies()).toEqual({'nonString':'1,2,3'}); + }); + + + it('should drop any null or undefined properties', function() { + scope.$cookies.nullVal = null; + scope.$cookies.undefVal = undefined; + scope.$eval(); + + expect(scope.$browser.cookies()).toEqual({}); + }); + + + it('should remove a cookie when a $cookies property is deleted', function() { + scope.$cookies.oatmealCookie = 'nom nom'; + scope.$eval(); + expect(scope.$browser.cookies()).toEqual({'oatmealCookie':'nom nom'}); + + delete scope.$cookies.oatmealCookie; + scope.$eval(); + + expect(scope.$browser.cookies()).toEqual({}); + }); + }); + + + describe('$sessionStore', function() { + + it('should serialize objects to json', function() { + scope.$sessionStore.put('objectCookie', {id: 123, name: 'blah'}); + scope.$eval(); + expect(scope.$browser.cookies()).toEqual({'objectCookie': '{"id":123,"name":"blah"}'}); + }); + + + it('should return all persisted items as a has via getAll', function() { + expect(scope.$sessionStore.getAll()).toEqual({}); + + scope.$sessionStore.put('object1', {id:1,foo:'bar1'}); + scope.$sessionStore.put('object2', {id:2,foo:'bar2'}); + + expect(scope.$sessionStore.getAll()).toEqual({'object1':{id:1,foo:'bar1'}, + 'object2':{id:2,foo:'bar2'}}); + }); + + + it('should deserialize json to object', function() { + scope.$browser.cookies('objectCookie', '{"id":123,"name":"blah"}'); + //TODO: This is a hacky way of calling the watch function, once pooling is refactored, this will go away. + scope.$browser.watches[1](scope.$browser.cookies()); + expect(scope.$sessionStore.get('objectCookie')).toEqual({id: 123, name: 'blah'}); + }); + + + it('should delete objects from the store when remove is called', function() { + scope.$sessionStore.put('gonner', { "I'll":"Be Back"}); + // TODO: Is this $eval necessary (why was it not here before?) + scope.$eval(); + expect(scope.$browser.cookies()).toEqual({'gonner': '{"I\'ll":"Be Back"}'}); + }); + + }); +});
\ No newline at end of file |
