/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import {ifEnvSupports, ifEnvSupportsWithDone, supportPatchXHROnProperty, zoneSymbol} from '../test-util'; declare const global: any; const wtfMock = global.wtfMock; describe('XMLHttpRequest', function() { let testZone: Zone; beforeEach(() => { testZone = Zone.current.fork({name: 'test'}); }); it('should intercept XHRs and treat them as MacroTasks', function(done) { let req: XMLHttpRequest; let onStable: any; const testZoneWithWtf = Zone.current.fork((Zone as any)['wtfZoneSpec']).fork({ name: 'TestZone', onHasTask: (delegate: ZoneDelegate, curr: Zone, target: Zone, hasTask: HasTaskState) => { if (!hasTask.macroTask) { onStable && onStable(); } } }); testZoneWithWtf.run(() => { req = new XMLHttpRequest(); const logs: string[] = []; req.onload = () => { logs.push('onload'); }; onStable = function() { expect(wtfMock.log[wtfMock.log.length - 2]) .toEqual('> Zone:invokeTask:XMLHttpRequest.send("::ProxyZone::WTF::TestZone")'); expect(wtfMock.log[wtfMock.log.length - 1]) .toEqual('< Zone:invokeTask:XMLHttpRequest.send'); if (supportPatchXHROnProperty()) { expect(wtfMock.log[wtfMock.log.length - 3]) .toMatch(/\< Zone\:invokeTask.*addEventListener\:load/); expect(wtfMock.log[wtfMock.log.length - 4]) .toMatch(/\> Zone\:invokeTask.*addEventListener\:load/); } // if browser can patch onload if ((req as any)[zoneSymbol('loadfalse')]) { expect(logs).toEqual(['onload']); } done(); }; req.open('get', '/', true); req.send(); const lastScheduled = wtfMock.log[wtfMock.log.length - 1]; expect(lastScheduled).toMatch('# Zone:schedule:macroTask:XMLHttpRequest.send'); }, null, undefined, 'unit-test'); }); it('should not trigger Zone callback of internal onreadystatechange', function(done) { const scheduleSpy = jasmine.createSpy('schedule'); const xhrZone = Zone.current.fork({ name: 'xhr', onScheduleTask: (delegate: ZoneDelegate, currentZone: Zone, targetZone, task: Task) => { if (task.type === 'eventTask') { scheduleSpy(task.source); } return delegate.scheduleTask(targetZone, task); } }); xhrZone.run(() => { const req = new XMLHttpRequest(); req.onload = function() { expect(Zone.current.name).toEqual('xhr'); if (supportPatchXHROnProperty()) { expect(scheduleSpy).toHaveBeenCalled(); } done(); }; req.open('get', '/', true); req.send(); }); }); it('should work with onreadystatechange', function(done) { let req: XMLHttpRequest; testZone.run(function() { req = new XMLHttpRequest(); req.onreadystatechange = function() { // Make sure that the wrapCallback will only be called once req.onreadystatechange = null as any; expect(Zone.current).toBe(testZone); done(); }; req.open('get', '/', true); }); req!.send(); }); it('should run onload listeners before internal readystatechange', function(done) { const logs: string[] = []; const xhrZone = Zone.current.fork({ name: 'xhr', onInvokeTask: (delegate, curr, target, task, applyThis, applyArgs) => { logs.push('invokeTask ' + task.source); return delegate.invokeTask(target, task, applyThis, applyArgs); } }); xhrZone.run(function() { const req = new XMLHttpRequest(); req.onload = function() { logs.push('onload'); (window as any)[Zone.__symbol__('setTimeout')](() => { expect(logs).toEqual([ 'invokeTask XMLHttpRequest.addEventListener:load', 'onload', 'invokeTask XMLHttpRequest.send' ]) done(); }); }; req.open('get', '/', true); req.send(); }); }); it('should invoke xhr task even onload listener throw error', function(done) { const oriWindowError = window.onerror; window.onerror = function() {}; try { const logs: string[] = []; const xhrZone = Zone.current.fork({ name: 'xhr', onInvokeTask: (delegate, curr, target, task, applyThis, applyArgs) => { logs.push('invokeTask ' + task.source); return delegate.invokeTask(target, task, applyThis, applyArgs); }, onHasTask: (delegate, curr, target, hasTaskState) => { if (hasTaskState.change === 'macroTask') { logs.push('hasTask ' + hasTaskState.macroTask); } return delegate.hasTask(target, hasTaskState); } }); xhrZone.run(function() { const req = new XMLHttpRequest(); req.onload = function() { logs.push('onload'); throw new Error('test'); }; const unhandledRejection = (e: PromiseRejectionEvent) => { logs.push(e.reason.message); }; window.addEventListener('unhandledrejection', unhandledRejection); req.addEventListener('load', () => { logs.push('onload1'); (window as any)[Zone.__symbol__('setTimeout')](() => { expect(logs).toEqual([ 'hasTask true', 'invokeTask XMLHttpRequest.addEventListener:load', 'onload', 'invokeTask XMLHttpRequest.addEventListener:load', 'onload1', 'invokeTask XMLHttpRequest.send', 'hasTask false', 'invokeTask Window.addEventListener:unhandledrejection', 'test' ]); window.removeEventListener('unhandledrejection', unhandledRejection); window.onerror = oriWindowError; done(); }); }); req.open('get', '/', true); req.send(); }); } catch (e: any) { window.onerror = oriWindowError; } }); it('should return null when access ontimeout first time without error', function() { let req: XMLHttpRequest = new XMLHttpRequest(); expect(req.ontimeout).toBe(null); }); const supportsOnProgress = function() { return 'onprogress' in (new XMLHttpRequest()); }; (supportsOnProgress).message = 'XMLHttpRequest.onprogress'; describe('onprogress', ifEnvSupports(supportsOnProgress, function() { it('should work with onprogress', function(done) { let req: XMLHttpRequest; testZone.run(function() { req = new XMLHttpRequest(); req.onprogress = function() { // Make sure that the wrapCallback will only be called once req.onprogress = null as any; expect(Zone.current).toBe(testZone); done(); }; req.open('get', '/', true); }); req!.send(); }); it('should allow canceling of an XMLHttpRequest', function(done) { const spy = jasmine.createSpy('spy'); let req: XMLHttpRequest; let pending = false; const trackingTestZone = Zone.current.fork({ name: 'tracking test zone', onHasTask: (delegate: ZoneDelegate, current: Zone, target: Zone, hasTaskState: HasTaskState) => { if (hasTaskState.change == 'macroTask') { pending = hasTaskState.macroTask; } delegate.hasTask(target, hasTaskState); } }); trackingTestZone.run(function() { req = new XMLHttpRequest(); req.onreadystatechange = function() { if (req.readyState === XMLHttpRequest.DONE) { if (req.status !== 0) { spy(); } } }; req.open('get', '/', true); req.send(); req.abort(); }); setTimeout(function() { expect(spy).not.toHaveBeenCalled(); expect(pending).toEqual(false); done(); }, 0); }); it('should allow aborting an XMLHttpRequest after its completed', function(done) { let req: XMLHttpRequest; testZone.run(function() { req = new XMLHttpRequest(); req.onreadystatechange = function() { if (req.readyState === XMLHttpRequest.DONE) { if (req.status !== 0) { setTimeout(function() { req.abort(); done(); }, 0); } } }; req.open('get', '/', true); req.send(); }); }); })); it('should preserve other setters', function() { const req = new XMLHttpRequest(); req.open('get', '/', true); req.send(); try { req.responseType = 'document'; expect(req.responseType).toBe('document'); } catch (e) { // Android browser: using this setter throws, this should be preserved expect(e.message).toBe('INVALID_STATE_ERR: DOM Exception 11'); } }); it('should work with synchronous XMLHttpRequest', function() { const log: HasTaskState[] = []; Zone.current .fork({ name: 'sync-xhr-test', onHasTask: function( delegate: ZoneDelegate, current: Zone, target: Zone, hasTaskState: HasTaskState) { log.push(hasTaskState); delegate.hasTask(target, hasTaskState); } }) .run(() => { const req = new XMLHttpRequest(); req.open('get', '/', false); req.send(); }); expect(log).toEqual([]); }); it('should preserve static constants', function() { expect(XMLHttpRequest.UNSENT).toEqual(0); expect(XMLHttpRequest.OPENED).toEqual(1); expect(XMLHttpRequest.HEADERS_RECEIVED).toEqual(2); expect(XMLHttpRequest.LOADING).toEqual(3); expect(XMLHttpRequest.DONE).toEqual(4); }); it('should work properly when send request multiple times on single xmlRequest instance', function(done) { testZone.run(function() { const req = new XMLHttpRequest(); req.open('get', '/', true); req.send(); req.onload = function() { req.onload = null as any; req.open('get', '/', true); req.onload = function() { done(); }; expect(() => { req.send(); }).not.toThrow(); }; }); }); it('should keep taskcount correctly when abort was called multiple times before request is done', function(done) { testZone.run(function() { const req = new XMLHttpRequest(); req.open('get', '/', true); req.send(); req.addEventListener('readystatechange', function(ev) { if (req.readyState >= 2) { expect(() => { req.abort(); }).not.toThrow(); done(); } }); }); }); it('should close xhr request if error happened when connect', function(done) { const logs: boolean[] = []; Zone.current .fork({ name: 'xhr', onHasTask: (delegate: ZoneDelegate, curr: Zone, target: Zone, taskState: HasTaskState) => { if (taskState.change === 'macroTask') { logs.push(taskState.macroTask); } return delegate.hasTask(target, taskState); } }) .run(function() { const req = new XMLHttpRequest(); req.open('get', 'http://notexists.url', true); req.send(); req.addEventListener('error', () => { expect(logs).toEqual([true, false]); done(); }); }); }); it('should trigger readystatechange if xhr request trigger cors error', (done) => { const req = new XMLHttpRequest(); let err: any = null; try { req.open('get', 'file:///test', true); } catch (err) { // in IE, open will throw Access is denied error done(); return; } req.addEventListener('readystatechange', function(ev) { if (req.readyState === 4) { const xhrScheduled = (req as any)[zoneSymbol('xhrScheduled')]; const task = (req as any)[zoneSymbol('xhrTask')]; if (xhrScheduled === false) { expect(task.state).toEqual('scheduling'); setTimeout(() => { if (err) { expect(task.state).toEqual('unknown'); } else { expect(task.state).toEqual('notScheduled'); } done(); }); } else { expect(task.state).toEqual('scheduled'); done(); } } }); try { req.send(); } catch (error) { err = error; } }); it('should invoke task if xhr request trigger cors error', (done) => { const logs: string[] = []; const zone = Zone.current.fork({ name: 'xhr', onHasTask: (delegate: ZoneDelegate, curr: Zone, target: Zone, hasTask: HasTaskState) => { logs.push(JSON.stringify(hasTask)); } }); const req = new XMLHttpRequest(); try { req.open('get', 'file:///test', true); } catch (err) { // in IE, open will throw Access is denied error done(); return; } zone.run(() => { let isError = false; let timerId = null; try { timerId = (window as any)[zoneSymbol('setTimeout')](() => { expect(logs).toEqual([ `{"microTask":false,"macroTask":true,"eventTask":false,"change":"macroTask"}`, `{"microTask":false,"macroTask":false,"eventTask":false,"change":"macroTask"}` ]); done(); }, 500); req.send(); } catch (error) { isError = true; (window as any)[zoneSymbol('clearTimeout')](timerId); done(); } }); }); it('should not throw error when get XMLHttpRequest.prototype.onreadystatechange the first time', function() { const func = function() { testZone.run(function() { const req = new XMLHttpRequest(); req.onreadystatechange; }); }; expect(func).not.toThrow(); }); it('should be in the zone when use XMLHttpRequest.addEventListener', function(done) { testZone.run(function() { // sometimes this case will cause timeout // so we set it longer const interval = (jasmine).DEFAULT_TIMEOUT_INTERVAL; (jasmine).DEFAULT_TIMEOUT_INTERVAL = 5000; const req = new XMLHttpRequest(); req.open('get', '/', true); req.addEventListener('readystatechange', function() { if (req.readyState === 4) { // expect(Zone.current.name).toEqual('test'); (jasmine).DEFAULT_TIMEOUT_INTERVAL = interval; done(); } }); req.send(); }); }); it('should return origin listener when call xhr.onreadystatechange', ifEnvSupportsWithDone(supportPatchXHROnProperty, function(done: Function) { testZone.run(function() { // sometimes this case will cause timeout // so we set it longer const req = new XMLHttpRequest(); req.open('get', '/', true); const interval = (jasmine).DEFAULT_TIMEOUT_INTERVAL; (jasmine).DEFAULT_TIMEOUT_INTERVAL = 5000; const listener = req.onreadystatechange = function() { if (req.readyState === 4) { (jasmine).DEFAULT_TIMEOUT_INTERVAL = interval; done(); } }; expect(req.onreadystatechange).toBe(listener); req.onreadystatechange = function() { return listener.call(this); }; req.send(); }); })); });