175 lines
6.6 KiB
TypeScript
175 lines
6.6 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright Google Inc. 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 {Injector} from '@angular/core';
|
|
import {AsyncTestCompleter, SpyObject, afterEach, beforeEach, describe, inject, it} from '@angular/core/testing/src/testing_internal';
|
|
import {BrowserJsonp} from '@angular/http/src/backends/browser_jsonp';
|
|
import {JSONPBackend, JSONPConnection} from '@angular/http/src/backends/jsonp_backend';
|
|
import {BaseRequestOptions, RequestOptions} from '@angular/http/src/base_request_options';
|
|
import {BaseResponseOptions, ResponseOptions} from '@angular/http/src/base_response_options';
|
|
import {ReadyState, RequestMethod, ResponseType} from '@angular/http/src/enums';
|
|
import {Request} from '@angular/http/src/static_request';
|
|
import {Response} from '@angular/http/src/static_response';
|
|
import {expect} from '@angular/platform-browser/testing/src/matchers';
|
|
|
|
let existingScripts: MockBrowserJsonp[] = [];
|
|
|
|
class MockBrowserJsonp extends BrowserJsonp {
|
|
src: string;
|
|
callbacks = new Map<string, (data: any) => any>();
|
|
|
|
addEventListener(type: string, cb: (data: any) => any) { this.callbacks.set(type, cb); }
|
|
|
|
removeEventListener(type: string, cb: Function) { this.callbacks.delete(type); }
|
|
|
|
dispatchEvent(type: string, argument: any = {}) {
|
|
const cb = this.callbacks.get(type);
|
|
if (cb) {
|
|
cb(argument);
|
|
}
|
|
}
|
|
|
|
build(url: string) {
|
|
const script = new MockBrowserJsonp();
|
|
script.src = url;
|
|
existingScripts.push(script);
|
|
return script;
|
|
}
|
|
|
|
send(node: any) { /* noop */
|
|
}
|
|
cleanup(node: any) { /* noop */
|
|
}
|
|
}
|
|
|
|
{
|
|
describe('JSONPBackend', () => {
|
|
let backend: JSONPBackend;
|
|
let sampleRequest: Request;
|
|
|
|
beforeEach(() => {
|
|
const injector = Injector.create([
|
|
{provide: ResponseOptions, useClass: BaseResponseOptions, deps: []},
|
|
{provide: BrowserJsonp, useClass: MockBrowserJsonp, deps: []},
|
|
{provide: JSONPBackend, useClass: JSONPBackend, deps: [BrowserJsonp, ResponseOptions]}
|
|
]);
|
|
backend = injector.get(JSONPBackend);
|
|
const base = new BaseRequestOptions();
|
|
sampleRequest =
|
|
new Request(base.merge(new RequestOptions({url: 'https://google.com'})) as any);
|
|
});
|
|
|
|
afterEach(() => { existingScripts = []; });
|
|
|
|
it('should create a connection', () => {
|
|
let instance: JSONPConnection = undefined !;
|
|
expect(() => instance = backend.createConnection(sampleRequest)).not.toThrow();
|
|
expect(instance).toBeAnInstanceOf(JSONPConnection);
|
|
});
|
|
|
|
|
|
describe('JSONPConnection', () => {
|
|
it('should use the injected BaseResponseOptions to create the response',
|
|
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
|
|
const connection = new (JSONPConnection as any)(
|
|
sampleRequest, new MockBrowserJsonp(),
|
|
new ResponseOptions({type: ResponseType.Error}));
|
|
connection.response.subscribe((res: Response) => {
|
|
expect(res.type).toBe(ResponseType.Error);
|
|
async.done();
|
|
});
|
|
connection.finished();
|
|
existingScripts[0].dispatchEvent('load');
|
|
}));
|
|
|
|
it('should ignore load/callback when disposed',
|
|
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
|
|
const connection = new (JSONPConnection as any)(sampleRequest, new MockBrowserJsonp());
|
|
const spy = new SpyObject();
|
|
const loadSpy = spy.spy('load');
|
|
const errorSpy = spy.spy('error');
|
|
const returnSpy = spy.spy('cancelled');
|
|
|
|
const request = connection.response.subscribe(loadSpy, errorSpy, returnSpy);
|
|
request.unsubscribe();
|
|
|
|
connection.finished('Fake data');
|
|
existingScripts[0].dispatchEvent('load');
|
|
|
|
setTimeout(() => {
|
|
expect(connection.readyState).toBe(ReadyState.Cancelled);
|
|
expect(loadSpy).not.toHaveBeenCalled();
|
|
expect(errorSpy).not.toHaveBeenCalled();
|
|
expect(returnSpy).not.toHaveBeenCalled();
|
|
async.done();
|
|
}, 10);
|
|
}));
|
|
|
|
it('should report error if loaded without invoking callback',
|
|
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
|
|
const connection = new (JSONPConnection as any)(sampleRequest, new MockBrowserJsonp());
|
|
connection.response.subscribe(
|
|
(res: Response) => {
|
|
expect('response listener called').toBe(false);
|
|
async.done();
|
|
},
|
|
(err: Response) => {
|
|
expect(err.text()).toEqual('JSONP injected script did not invoke callback.');
|
|
async.done();
|
|
});
|
|
|
|
existingScripts[0].dispatchEvent('load');
|
|
}));
|
|
|
|
it('should report error if script contains error',
|
|
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
|
|
const connection = new (JSONPConnection as any)(sampleRequest, new MockBrowserJsonp());
|
|
|
|
connection.response.subscribe(
|
|
(res: Response) => {
|
|
expect('response listener called').toBe(false);
|
|
async.done();
|
|
},
|
|
(err: Response) => {
|
|
expect(err.text()).toBe('Oops!');
|
|
async.done();
|
|
});
|
|
|
|
existingScripts[0].dispatchEvent('error', ({message: 'Oops!'}));
|
|
}));
|
|
|
|
it('should throw if request method is not GET', () => {
|
|
[RequestMethod.Post, RequestMethod.Put, RequestMethod.Delete, RequestMethod.Options,
|
|
RequestMethod.Head, RequestMethod.Patch]
|
|
.forEach(method => {
|
|
const base = new BaseRequestOptions();
|
|
const req = new Request(base.merge(
|
|
new RequestOptions({url: 'https://google.com', method: method})) as any);
|
|
expect(
|
|
() => new (JSONPConnection as any)(req, new MockBrowserJsonp())
|
|
.response.subscribe())
|
|
.toThrowError();
|
|
});
|
|
});
|
|
|
|
it('should respond with data passed to callback',
|
|
inject([AsyncTestCompleter], (async: AsyncTestCompleter) => {
|
|
const connection = new (JSONPConnection as any)(sampleRequest, new MockBrowserJsonp());
|
|
|
|
connection.response.subscribe((res: Response) => {
|
|
expect(res.json()).toEqual(({fake_payload: true, blob_id: 12345}));
|
|
async.done();
|
|
});
|
|
|
|
connection.finished(({fake_payload: true, blob_id: 12345}));
|
|
existingScripts[0].dispatchEvent('load');
|
|
}));
|
|
});
|
|
});
|
|
}
|