mirror of
https://github.com/Hopiu/angular.js.git
synced 2026-03-17 07:40:22 +00:00
the max size for safari cookies has changed sligtly so I had to adjust the test to make cookie creation fail on this browser
643 lines
19 KiB
JavaScript
643 lines
19 KiB
JavaScript
'use strict';
|
|
|
|
describe('browser', function(){
|
|
|
|
var browser, fakeWindow, xhr, logs, scripts, removedScripts, setTimeoutQueue;
|
|
|
|
function fakeSetTimeout(fn) {
|
|
return setTimeoutQueue.push(fn) - 1; //return position in the queue
|
|
}
|
|
|
|
function fakeClearTimeout(deferId) {
|
|
setTimeoutQueue[deferId] = noop; //replace fn with noop to preserve other deferId indexes
|
|
}
|
|
|
|
fakeSetTimeout.flush = function() {
|
|
var currentTimeoutQueue = setTimeoutQueue;
|
|
setTimeoutQueue = [];
|
|
forEach(currentTimeoutQueue, function(fn) {
|
|
fn();
|
|
});
|
|
};
|
|
|
|
|
|
beforeEach(function(){
|
|
setTimeoutQueue = [];
|
|
scripts = [];
|
|
removedScripts = [];
|
|
xhr = null;
|
|
fakeWindow = {
|
|
location: {href:"http://server"},
|
|
setTimeout: fakeSetTimeout,
|
|
clearTimeout: fakeClearTimeout
|
|
};
|
|
|
|
var fakeBody = [{appendChild: function(node){scripts.push(node);},
|
|
removeChild: function(node){removedScripts.push(node);}}];
|
|
|
|
var FakeXhr = function(){
|
|
xhr = this;
|
|
this.open = function(method, url, async){
|
|
xhr.method = method;
|
|
xhr.url = url;
|
|
xhr.async = async;
|
|
xhr.headers = {};
|
|
};
|
|
this.setRequestHeader = function(key, value){
|
|
xhr.headers[key] = value;
|
|
};
|
|
this.send = function(post){
|
|
xhr.post = post;
|
|
};
|
|
this.getResponseHeader = function(header) {
|
|
return header;
|
|
};
|
|
this.getAllResponseHeaders = function() {
|
|
return 'Content-Type: application/json\n\rContent-Encoding: gzip\n\rContent-Type: text/json';
|
|
}
|
|
|
|
};
|
|
|
|
logs = {log:[], warn:[], info:[], error:[]};
|
|
|
|
var fakeLog = {log: function() { logs.log.push(slice.call(arguments)); },
|
|
warn: function() { logs.warn.push(slice.call(arguments)); },
|
|
info: function() { logs.info.push(slice.call(arguments)); },
|
|
error: function() { logs.error.push(slice.call(arguments)); }};
|
|
|
|
browser = new Browser(fakeWindow, jqLite(window.document), fakeBody, FakeXhr,
|
|
fakeLog);
|
|
});
|
|
|
|
it('should contain cookie cruncher', function() {
|
|
expect(browser.cookies).toBeDefined();
|
|
});
|
|
|
|
describe('outstading requests', function(){
|
|
it('should process callbacks immedietly with no outstanding requests', function(){
|
|
var callback = jasmine.createSpy('callback');
|
|
browser.notifyWhenNoOutstandingRequests(callback);
|
|
expect(callback).toHaveBeenCalled();
|
|
});
|
|
|
|
it('should queue callbacks with outstanding requests', function(){
|
|
var callback = jasmine.createSpy('callback');
|
|
browser.xhr('GET', '/url', null, noop);
|
|
browser.notifyWhenNoOutstandingRequests(callback);
|
|
expect(callback).not.toHaveBeenCalled();
|
|
|
|
xhr.readyState = 4;
|
|
xhr.onreadystatechange();
|
|
expect(callback).toHaveBeenCalled();
|
|
});
|
|
});
|
|
|
|
describe('xhr', function(){
|
|
describe('JSON', function(){
|
|
var log;
|
|
|
|
function callback(code, data) {
|
|
log += code + ':' + data + ';';
|
|
}
|
|
|
|
beforeEach(function() {
|
|
log = "";
|
|
});
|
|
|
|
|
|
// We don't have unit tests for IE because script.readyState is readOnly.
|
|
// Instead we run e2e tests on all browsers - see e2e for $xhr.
|
|
if (!msie) {
|
|
|
|
it('should add script tag for JSONP request', function() {
|
|
var notify = jasmine.createSpy('notify');
|
|
browser.xhr('JSON', 'http://example.org/path?cb=JSON_CALLBACK', null, callback);
|
|
browser.notifyWhenNoOutstandingRequests(notify);
|
|
expect(notify).not.toHaveBeenCalled();
|
|
expect(scripts.length).toEqual(1);
|
|
var script = scripts[0];
|
|
var url = script.src.split('?cb=');
|
|
expect(url[0]).toEqual('http://example.org/path');
|
|
expect(typeof fakeWindow[url[1]]).toEqual($function);
|
|
fakeWindow[url[1]]('data');
|
|
script.onload();
|
|
|
|
expect(notify).toHaveBeenCalled();
|
|
expect(log).toEqual('200:data;');
|
|
expect(scripts).toEqual(removedScripts);
|
|
expect(fakeWindow[url[1]]).toBeUndefined();
|
|
});
|
|
|
|
|
|
it('should call callback when script fails to load', function() {
|
|
browser.xhr('JSON', 'http://example.org/path?cb=JSON_CALLBACK', null, callback);
|
|
var script = scripts[0];
|
|
expect(typeof script.onload).toBe($function);
|
|
expect(typeof script.onerror).toBe($function);
|
|
script.onerror();
|
|
|
|
expect(log).toEqual('undefined:undefined;');
|
|
});
|
|
|
|
|
|
it('should update the outstandingRequests counter for successful requests', function() {
|
|
var notify = jasmine.createSpy('notify');
|
|
browser.xhr('JSON', 'http://example.org/path?cb=JSON_CALLBACK', null, callback);
|
|
browser.notifyWhenNoOutstandingRequests(notify);
|
|
expect(notify).not.toHaveBeenCalled();
|
|
|
|
var script = scripts[0];
|
|
var url = script.src.split('?cb=');
|
|
fakeWindow[url[1]]('data');
|
|
script.onload();
|
|
|
|
expect(notify).toHaveBeenCalled();
|
|
});
|
|
|
|
|
|
it('should update the outstandingRequests counter for failed requests', function() {
|
|
var notify = jasmine.createSpy('notify');
|
|
browser.xhr('JSON', 'http://example.org/path?cb=JSON_CALLBACK', null, callback);
|
|
browser.notifyWhenNoOutstandingRequests(notify);
|
|
expect(notify).not.toHaveBeenCalled();
|
|
|
|
scripts[0].onerror();
|
|
|
|
expect(notify).toHaveBeenCalled();
|
|
});
|
|
}
|
|
});
|
|
|
|
|
|
it('should normalize IE\'s 1223 status code into 204', function() {
|
|
var callback = jasmine.createSpy('XHR');
|
|
|
|
browser.xhr('GET', 'URL', 'POST', callback);
|
|
|
|
xhr.status = 1223;
|
|
xhr.readyState = 4;
|
|
xhr.onreadystatechange();
|
|
|
|
expect(callback).toHaveBeenCalled();
|
|
expect(callback.argsForCall[0][0]).toEqual(204);
|
|
});
|
|
|
|
it('should set only the requested headers', function() {
|
|
var code, response, headers = {};
|
|
browser.xhr('POST', 'URL', null, function(c,r){
|
|
code = c;
|
|
response = r;
|
|
}, {'X-header1': 'value1', 'X-header2': 'value2'});
|
|
|
|
expect(xhr.method).toEqual('POST');
|
|
expect(xhr.url).toEqual('URL');
|
|
expect(xhr.post).toEqual('');
|
|
expect(xhr.headers).toEqual({
|
|
"X-header1":"value1",
|
|
"X-header2":"value2"
|
|
});
|
|
|
|
xhr.status = 202;
|
|
xhr.responseText = 'RESPONSE';
|
|
xhr.readyState = 4;
|
|
xhr.onreadystatechange();
|
|
|
|
expect(code).toEqual(202);
|
|
expect(response).toEqual('RESPONSE');
|
|
});
|
|
|
|
describe('response headers', function() {
|
|
it('should return a single response header', function() {
|
|
var headerA;
|
|
|
|
browser.xhr('GET', 'URL', null, function(code, resp, headers) {
|
|
headerA = headers('A-Header');
|
|
});
|
|
|
|
xhr.status = 200;
|
|
xhr.responseText = 'RESPONSE';
|
|
xhr.readyState = 4;
|
|
xhr.onreadystatechange();
|
|
|
|
expect(headerA).toEqual('a-header');
|
|
});
|
|
|
|
it('should return an object containing all response headers', function() {
|
|
var allHeaders;
|
|
|
|
browser.xhr('GET', 'URL', null, function(code, resp, headers) {
|
|
allHeaders = headers();
|
|
});
|
|
|
|
xhr.status = 200;
|
|
xhr.responseText = 'RESPONSE';
|
|
xhr.readyState = 4;
|
|
xhr.onreadystatechange();
|
|
|
|
expect(allHeaders).toEqual({
|
|
'content-type': 'application/json, text/json',
|
|
'content-encoding': 'gzip'
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('defer', function() {
|
|
it('should execute fn asynchroniously via setTimeout', function() {
|
|
var counter = 0;
|
|
browser.defer(function() {counter++;});
|
|
expect(counter).toBe(0);
|
|
|
|
fakeSetTimeout.flush();
|
|
expect(counter).toBe(1);
|
|
});
|
|
|
|
|
|
it('should update outstandingRequests counter', function() {
|
|
var callback = jasmine.createSpy('callback');
|
|
browser.defer(callback);
|
|
expect(callback).not.toHaveBeenCalled();
|
|
|
|
fakeSetTimeout.flush();
|
|
expect(callback).toHaveBeenCalled();
|
|
});
|
|
|
|
|
|
it('should return unique deferId', function() {
|
|
var deferId1 = browser.defer(noop),
|
|
deferId2 = browser.defer(noop);
|
|
|
|
expect(deferId1).toBeDefined();
|
|
expect(deferId2).toBeDefined();
|
|
expect(deferId1).not.toEqual(deferId2);
|
|
});
|
|
|
|
|
|
describe('cancel', function() {
|
|
it('should allow tasks to be canceled with returned deferId', function() {
|
|
var log = [],
|
|
deferId1 = browser.defer(function() { log.push('cancel me'); }),
|
|
deferId2 = browser.defer(function() { log.push('ok'); }),
|
|
deferId3 = browser.defer(function() { log.push('cancel me, now!'); });
|
|
|
|
expect(log).toEqual([]);
|
|
browser.defer.cancel(deferId1);
|
|
browser.defer.cancel(deferId3);
|
|
fakeSetTimeout.flush();
|
|
expect(log).toEqual(['ok']);
|
|
});
|
|
});
|
|
});
|
|
|
|
|
|
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";
|
|
}
|
|
}
|
|
|
|
beforeEach(function() {
|
|
deleteAllCookies();
|
|
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 cookies(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 cookies(cookieName, string)', function() {
|
|
|
|
it('should create and store a cookie', function() {
|
|
browser.cookies('cookieName', 'cookie=Value');
|
|
expect(document.cookie).toMatch(/cookieName=cookie%3DValue;? ?/);
|
|
expect(browser.cookies()).toEqual({'cookieName':'cookie=Value'});
|
|
});
|
|
|
|
|
|
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');
|
|
});
|
|
|
|
it('should log warnings when 4kb per cookie storage limit is reached', function() {
|
|
var i, longVal = '', cookieStr;
|
|
|
|
for(i=0; i<4091; i++) {
|
|
longVal += '+';
|
|
}
|
|
|
|
cookieStr = document.cookie;
|
|
browser.cookies('x', longVal); //total size 4093-4096, so it should go through
|
|
expect(document.cookie).not.toEqual(cookieStr);
|
|
expect(browser.cookies()['x']).toEqual(longVal);
|
|
expect(logs.warn).toEqual([]);
|
|
|
|
browser.cookies('x', longVal + 'xxxx'); //total size 4097-4099, a warning should be logged
|
|
expect(logs.warn).toEqual(
|
|
[[ "Cookie 'x' possibly not set or overflowed because it was too large (4097 > 4096 " +
|
|
"bytes)!" ]]);
|
|
|
|
//force browser to dropped a cookie and make sure that the cache is not out of sync
|
|
browser.cookies('x', 'shortVal');
|
|
expect(browser.cookies().x).toEqual('shortVal'); //needed to prime the cache
|
|
cookieStr = document.cookie;
|
|
browser.cookies('x', longVal + longVal + longVal); //should be too long for all browsers
|
|
|
|
if (document.cookie !== cookieStr) {
|
|
fail("browser didn't drop long cookie when it was expected. make the cookie in this " +
|
|
"test longer");
|
|
}
|
|
|
|
expect(browser.cookies().x).toEqual('shortVal');
|
|
});
|
|
|
|
it('should log warnings when 20 cookies per domain storage limit is reached', function() {
|
|
var i, str, cookieStr;
|
|
|
|
for (i=0; i<20; i++) {
|
|
str = '' + i;
|
|
browser.cookies(str, str);
|
|
}
|
|
|
|
i=0;
|
|
for (str in browser.cookies()) {
|
|
i++;
|
|
}
|
|
expect(i).toEqual(20);
|
|
expect(logs.warn).toEqual([]);
|
|
cookieStr = document.cookie;
|
|
|
|
browser.cookies('one', 'more');
|
|
expect(logs.warn).toEqual([]);
|
|
|
|
//if browser dropped a cookie (very likely), make sure that the cache is not out of sync
|
|
if (document.cookie === cookieStr) {
|
|
expect(size(browser.cookies())).toEqual(20);
|
|
} else {
|
|
expect(size(browser.cookies())).toEqual(21);
|
|
}
|
|
});
|
|
});
|
|
|
|
|
|
describe('get via cookies()[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=baz";
|
|
expect(browser.cookies().foo).toEqual('bar=baz');
|
|
});
|
|
|
|
|
|
it ('should unescape cookie values that were escaped by puts', function() {
|
|
document.cookie = "cookie2%3Dbar%3Bbaz=val%3Due";
|
|
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 via cookies()', 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({});
|
|
});
|
|
});
|
|
|
|
|
|
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';
|
|
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'});
|
|
});
|
|
|
|
});
|
|
|
|
describe('poller', function(){
|
|
|
|
it('should call functions in pollFns in regular intervals', function(){
|
|
var log = '';
|
|
browser.addPollFn(function(){log+='a';});
|
|
browser.addPollFn(function(){log+='b';});
|
|
expect(log).toEqual('');
|
|
fakeSetTimeout.flush();
|
|
expect(log).toEqual('ab');
|
|
fakeSetTimeout.flush();
|
|
expect(log).toEqual('abab');
|
|
});
|
|
|
|
it('should startPoller', function(){
|
|
expect(setTimeoutQueue.length).toEqual(0);
|
|
|
|
browser.addPollFn(function(){});
|
|
expect(setTimeoutQueue.length).toEqual(1);
|
|
|
|
//should remain 1 as it is the check fn
|
|
browser.addPollFn(function(){});
|
|
expect(setTimeoutQueue.length).toEqual(1);
|
|
});
|
|
|
|
it('should return fn that was passed into addPollFn', function() {
|
|
var fn = function() { return 1; };
|
|
var returnedFn = browser.addPollFn(fn);
|
|
expect(returnedFn).toBe(fn);
|
|
});
|
|
});
|
|
|
|
|
|
describe('url api', function() {
|
|
it('should use $browser poller to detect url changes when onhashchange event is unsupported',
|
|
function() {
|
|
|
|
fakeWindow = {
|
|
location: {href:"http://server"},
|
|
document: {},
|
|
setTimeout: fakeSetTimeout
|
|
};
|
|
|
|
browser = new Browser(fakeWindow, {}, {});
|
|
browser.startPoller = function() {};
|
|
|
|
var events = [];
|
|
|
|
browser.onHashChange(function() {
|
|
events.push('x');
|
|
});
|
|
|
|
fakeWindow.location.href = "http://server/#newHash";
|
|
expect(events).toEqual([]);
|
|
fakeSetTimeout.flush();
|
|
expect(events).toEqual(['x']);
|
|
|
|
//don't do anything if url hasn't changed
|
|
events = [];
|
|
fakeSetTimeout.flush();
|
|
expect(events).toEqual([]);
|
|
});
|
|
|
|
|
|
it('should use onhashchange events to detect url changes when supported by browser',
|
|
function() {
|
|
|
|
var onHashChngListener;
|
|
|
|
fakeWindow = {location: {href:"http://server"},
|
|
addEventListener: function(type, listener) {
|
|
expect(type).toEqual('hashchange');
|
|
onHashChngListener = listener;
|
|
},
|
|
attachEvent: function(type, listener) {
|
|
expect(type).toEqual('onhashchange');
|
|
onHashChngListener = listener;
|
|
},
|
|
removeEventListener: angular.noop,
|
|
detachEvent: angular.noop,
|
|
document: {}
|
|
};
|
|
fakeWindow.onhashchange = true;
|
|
|
|
browser = new Browser(fakeWindow, {}, {});
|
|
|
|
var events = [],
|
|
event = {type: "hashchange"};
|
|
|
|
browser.onHashChange(function(e) {
|
|
events.push(e);
|
|
});
|
|
|
|
expect(events).toEqual([]);
|
|
onHashChngListener(event);
|
|
|
|
expect(events.length).toBe(1);
|
|
expect(events[0].originalEvent || events[0]).toBe(event); // please jQuery and jqLite
|
|
|
|
// clean up the jqLite cache so that the global afterEach doesn't complain
|
|
if (!jQuery) {
|
|
jqLite(fakeWindow).dealoc();
|
|
}
|
|
});
|
|
|
|
// asynchronous test
|
|
it('should fire onHashChange when location.hash change', function() {
|
|
var callback = jasmine.createSpy('onHashChange');
|
|
browser = new Browser(window, {}, {});
|
|
browser.onHashChange(callback);
|
|
|
|
window.location.hash = 'new-hash';
|
|
browser.addPollFn(function() {});
|
|
|
|
waitsFor(function() {
|
|
return callback.callCount;
|
|
}, 'onHashChange callback to be called', 1000);
|
|
|
|
runs(function() {
|
|
if (!jQuery) jqLite(window).dealoc();
|
|
window.location.hash = '';
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('addJs', function() {
|
|
|
|
it('should append a script tag to body', function() {
|
|
browser.addJs('http://localhost/bar.js');
|
|
expect(scripts.length).toBe(1);
|
|
expect(scripts[0].src).toBe('http://localhost/bar.js');
|
|
expect(scripts[0].id).toBe('');
|
|
});
|
|
|
|
|
|
it('should append a script with an id to body', function() {
|
|
browser.addJs('http://localhost/bar.js', 'foo-id');
|
|
expect(scripts.length).toBe(1);
|
|
expect(scripts[0].src).toBe('http://localhost/bar.js');
|
|
expect(scripts[0].id).toBe('foo-id');
|
|
});
|
|
|
|
|
|
it('should return the appended script element', function() {
|
|
var script = browser.addJs('http://localhost/bar.js');
|
|
expect(script).toBe(scripts[0]);
|
|
});
|
|
});
|
|
});
|