/** * Copyright 2018 Google Inc. All rights reserved. * Modifications copyright (c) Microsoft Corporation. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { EventEmitter } from 'events'; import { CRSession, CRConnection } from '../crConnection'; import { debugError } from '../../helper'; import { Protocol } from '../protocol'; import { Events } from '../events'; import * as types from '../../types'; import * as js from '../../javascript'; import * as console from '../../console'; import { CRExecutionContext } from '../crExecutionContext'; import { toConsoleMessageLocation, exceptionToError } from '../crProtocolHelper'; import { ChromiumPage } from '../crPage'; type AddToConsoleCallback = (type: string, args: js.JSHandle[], location: console.ConsoleMessageLocation) => void; type HandleExceptionCallback = (error: Error) => void; export class CRWorkers { private _workers = new Map(); constructor(client: CRSession, page: ChromiumPage, addToConsole: AddToConsoleCallback, handleException: HandleExceptionCallback) { client.on('Target.attachedToTarget', event => { if (event.targetInfo.type !== 'worker') return; const session = CRConnection.fromSession(client).session(event.sessionId); const worker = new CRWorker(session, event.targetInfo.url, addToConsole, handleException); this._workers.set(event.sessionId, worker); page.emit(Events.CRPage.WorkerCreated, worker); }); client.on('Target.detachedFromTarget', event => { const worker = this._workers.get(event.sessionId); if (!worker) return; page.emit(Events.CRPage.WorkerDestroyed, worker); this._workers.delete(event.sessionId); }); } list(): CRWorker[] { return Array.from(this._workers.values()); } } export class CRWorker extends EventEmitter { private _client: CRSession; private _url: string; private _executionContextPromise: Promise; private _executionContextCallback: (value?: js.ExecutionContext) => void; constructor(client: CRSession, url: string, addToConsole: AddToConsoleCallback, handleException: HandleExceptionCallback) { super(); this._client = client; this._url = url; this._executionContextPromise = new Promise(x => this._executionContextCallback = x); let jsHandleFactory: (o: Protocol.Runtime.RemoteObject) => js.JSHandle; this._client.once('Runtime.executionContextCreated', async event => { jsHandleFactory = remoteObject => executionContext._createHandle(remoteObject); const executionContext = new js.ExecutionContext(new CRExecutionContext(client, event.context)); this._executionContextCallback(executionContext); }); // This might fail if the target is closed before we recieve all execution contexts. this._client.send('Runtime.enable', {}).catch(debugError); this._client.on('Runtime.consoleAPICalled', event => addToConsole(event.type, event.args.map(jsHandleFactory), toConsoleMessageLocation(event.stackTrace))); this._client.on('Runtime.exceptionThrown', exception => handleException(exceptionToError(exception.exceptionDetails))); } url(): string { return this._url; } evaluate: types.Evaluate = async (pageFunction, ...args) => { return (await this._executionContextPromise).evaluate(pageFunction, ...args as any); } evaluateHandle: types.EvaluateHandle = async (pageFunction, ...args) => { return (await this._executionContextPromise).evaluateHandle(pageFunction, ...args as any); } }