/** * @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 {Subject} from 'rxjs'; import {Adapter, Context} from '../src/adapter'; import {AssetGroupConfig, Manifest} from '../src/manifest'; import {sha1} from '../src/sha1'; import {MockCacheStorage} from './cache'; import {MockHeaders, MockRequest, MockResponse} from './fetch'; import {MockServerState, MockServerStateBuilder} from './mock'; import {normalizeUrl, parseUrl} from './utils'; const EMPTY_SERVER_STATE = new MockServerStateBuilder().build(); export class MockClient { queue = new Subject(); constructor(readonly id: string) {} readonly messages: Object[] = []; postMessage(message: Object): void { this.messages.push(message); this.queue.next(message); } } export class SwTestHarnessBuilder { private origin = parseUrl(this.scopeUrl).origin; private server = EMPTY_SERVER_STATE; private caches = new MockCacheStorage(this.origin); constructor(private scopeUrl = 'http://localhost/') {} withCacheState(cache: string): SwTestHarnessBuilder { this.caches = new MockCacheStorage(this.origin, cache); return this; } withServerState(state: MockServerState): SwTestHarnessBuilder { this.server = state; return this; } build(): SwTestHarness { return new SwTestHarness(this.server, this.caches, this.scopeUrl); } } export class MockClients implements Clients { private clients = new Map(); add(clientId: string): void { if (this.clients.has(clientId)) { return; } this.clients.set(clientId, new MockClient(clientId)); } remove(clientId: string): void { this.clients.delete(clientId); } async get(id: string): Promise { return this.clients.get(id)! as any as Client; } getMock(id: string): MockClient|undefined { return this.clients.get(id); } async matchAll(): Promise { return Array.from(this.clients.values()) as any[] as Client[]; } async claim(): Promise {} } export class SwTestHarness extends Adapter implements ServiceWorkerGlobalScope, Context { readonly clients = new MockClients(); private eventHandlers = new Map(); private skippedWaiting = true; private selfMessageQueue: any[] = []; autoAdvanceTime = false; // TODO(issue/24571): remove '!'. unregistered!: boolean; readonly notifications: {title: string, options: Object}[] = []; readonly registration: ServiceWorkerRegistration = { active: { postMessage: (msg: any) => { this.selfMessageQueue.push(msg); }, }, scope: this.scopeUrl, showNotification: (title: string, options: Object) => { this.notifications.push({title, options}); }, unregister: () => { this.unregistered = true; }, } as any; static envIsSupported(): boolean { if (typeof URL === 'function') { return true; } // If we're in a browser that doesn't support URL at this point, don't go any further // since browser builds use requirejs which will fail on the `require` call below. if (typeof window !== 'undefined' && window) { return false; } // In older Node.js versions, the `URL` global does not exist. We can use `url` instead. const url = (typeof require === 'function') && require('url'); return url && (typeof url.parse === 'function') && (typeof url.resolve === 'function'); } get time() { return this.mockTime; } private mockTime = Date.now(); private timers: { at: number, duration: number, fn: Function, fired: boolean, }[] = []; parseUrl = parseUrl; constructor( private server: MockServerState, readonly caches: MockCacheStorage, scopeUrl: string) { super(scopeUrl); } async resolveSelfMessages(): Promise { while (this.selfMessageQueue.length > 0) { const queue = this.selfMessageQueue; this.selfMessageQueue = []; await queue.reduce(async (previous, msg) => { await previous; await this.handleMessage(msg, null); }, Promise.resolve()); } } async startup(firstTime: boolean = false): Promise { if (!firstTime) { return null; } let skippedWaiting: boolean = false; if (this.eventHandlers.has('install')) { const installEvent = new MockInstallEvent(); this.eventHandlers.get('install')!(installEvent); await installEvent.ready; skippedWaiting = this.skippedWaiting; } if (this.eventHandlers.has('activate')) { const activateEvent = new MockActivateEvent(); this.eventHandlers.get('activate')!(activateEvent); await activateEvent.ready; } return skippedWaiting; } updateServerState(server?: MockServerState): void { this.server = server || EMPTY_SERVER_STATE; } fetch(req: RequestInfo): Promise { if (typeof req === 'string') { return this.server.fetch(new MockRequest(normalizeUrl(req, this.scopeUrl))); } else { const mockReq = req.clone() as MockRequest; mockReq.url = normalizeUrl(mockReq.url, this.scopeUrl); return this.server.fetch(mockReq); } } addEventListener(event: string, handler: Function): void { this.eventHandlers.set(event, handler); } removeEventListener(event: string, handler?: Function): void { this.eventHandlers.delete(event); } newRequest(url: string, init: Object = {}): Request { return new MockRequest(normalizeUrl(url, this.scopeUrl), init); } newResponse(body: string, init: Object = {}): Response { return new MockResponse(body, init); } newHeaders(headers: {[name: string]: string}): Headers { return Object.keys(headers).reduce((mock, name) => { mock.set(name, headers[name]); return mock; }, new MockHeaders()); } async skipWaiting(): Promise { this.skippedWaiting = true; } waitUntil(promise: Promise): void {} handleFetch(req: Request, clientId: string|null = null): [Promise, Promise] { if (!this.eventHandlers.has('fetch')) { throw new Error('No fetch handler registered'); } const event = new MockFetchEvent(req, clientId); this.eventHandlers.get('fetch')!.call(this, event); if (clientId) { this.clients.add(clientId); } return [event.response, event.ready]; } handleMessage(data: Object, clientId: string|null): Promise { if (!this.eventHandlers.has('message')) { throw new Error('No message handler registered'); } let event: MockMessageEvent; if (clientId === null) { event = new MockMessageEvent(data, null); } else { this.clients.add(clientId); event = new MockMessageEvent(data, this.clients.getMock(clientId) || null); } this.eventHandlers.get('message')!.call(this, event); return event.ready; } handlePush(data: Object): Promise { if (!this.eventHandlers.has('push')) { throw new Error('No push handler registered'); } const event = new MockPushEvent(data); this.eventHandlers.get('push')!.call(this, event); return event.ready; } handleClick(notification: Object, action?: string): Promise { if (!this.eventHandlers.has('notificationclick')) { throw new Error('No notificationclick handler registered'); } const event = new MockNotificationEvent(notification, action); this.eventHandlers.get('notificationclick')!.call(this, event); return event.ready; } timeout(ms: number): Promise { const promise = new Promise(resolve => { this.timers.push({ at: this.mockTime + ms, duration: ms, fn: resolve, fired: false, }); }); if (this.autoAdvanceTime) { this.advance(ms); } return promise; } advance(by: number): void { this.mockTime += by; this.timers.filter(timer => !timer.fired) .filter(timer => timer.at <= this.mockTime) .forEach(timer => { timer.fired = true; timer.fn(); }); } isClient(obj: any): obj is Client { return obj instanceof MockClient; } } interface StaticFile { url: string; contents: string; hash?: string; } export class AssetGroupBuilder { constructor(private up: ConfigBuilder, readonly name: string) {} private files: StaticFile[] = []; addFile(url: string, contents: string, hashed: boolean = true): AssetGroupBuilder { const file: StaticFile = {url, contents, hash: undefined}; if (hashed) { file.hash = sha1(contents); } this.files.push(file); return this; } finish(): ConfigBuilder { return this.up; } toManifestGroup(): AssetGroupConfig { return null!; } } export class ConfigBuilder { assetGroups = new Map(); addAssetGroup(name: string): ConfigBuilder { const builder = new AssetGroupBuilder(this, name); this.assetGroups.set(name, builder); return this; } finish(): Manifest { const assetGroups = Array.from(this.assetGroups.values()).map(group => group.toManifestGroup()); const hashTable = {}; return { configVersion: 1, timestamp: 1234567890123, index: '/index.html', assetGroups, navigationUrls: [], hashTable, }; } } class OneTimeContext implements Context { private queue: Promise[] = []; waitUntil(promise: Promise): void { this.queue.push(promise); } get ready(): Promise { return (async () => { while (this.queue.length > 0) { await this.queue.shift(); } })(); } } class MockExtendableEvent extends OneTimeContext {} class MockFetchEvent extends MockExtendableEvent { response: Promise = Promise.resolve(undefined); constructor(readonly request: Request, readonly clientId: string|null) { super(); } respondWith(promise: Promise): Promise { this.response = promise; return promise; } } class MockMessageEvent extends MockExtendableEvent { constructor(readonly data: Object, readonly source: MockClient|null) { super(); } } class MockPushEvent extends MockExtendableEvent { constructor(private _data: Object) { super(); } data = { json: () => this._data, }; } class MockNotificationEvent extends MockExtendableEvent { constructor(private _notification: any, readonly action?: string) { super(); } readonly notification = {...this._notification, close: () => undefined}; } class MockInstallEvent extends MockExtendableEvent {} class MockActivateEvent extends MockExtendableEvent {}