168 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			168 lines
		
	
	
		
			6.5 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 {
 | |
|   // TODO(issue/24571): remove '!'.
 | |
|   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(
 | |
|                () => async.fail('Response listener should not be called'), (err: Response) => {
 | |
|                  expect(err.text()).toBe('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(
 | |
|                () => async.fail('Response listener should not be called'), (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');
 | |
|          }));
 | |
|     });
 | |
|   });
 | |
| }
 |