From 292216652b0e1771564718bf6cc1e1139446316a Mon Sep 17 00:00:00 2001 From: jrandolf <101637635+jrandolf@users.noreply.github.com> Date: Wed, 31 Aug 2022 10:50:22 +0200 Subject: [PATCH] chore: add injection framework (#8862) * chore: add injection framework --- src/common/Frame.ts | 2 +- src/common/IsolatedWorld.ts | 19 ++++++++-------- src/injected/injected.ts | 13 ++++++++++- src/injected/util.ts | 18 +++++++++++++++ src/templates/injected.ts.tmpl | 9 ++++++++ src/tsconfig.esm.json | 3 ++- test/src/injected.spec.ts | 40 ++++++++++++++++++++++++++++++++++ test/src/network.spec.ts | 9 +++++--- utils/generate_sources.ts | 24 ++++++++++++++------ utils/internal/job.ts | 4 ++-- 10 files changed, 116 insertions(+), 25 deletions(-) create mode 100644 src/injected/util.ts create mode 100644 src/templates/injected.ts.tmpl create mode 100644 test/src/injected.spec.ts diff --git a/src/common/Frame.ts b/src/common/Frame.ts index 2714157a..5eb99cc1 100644 --- a/src/common/Frame.ts +++ b/src/common/Frame.ts @@ -214,7 +214,7 @@ export class Frame { this.#client = client; this.worlds = { [MAIN_WORLD]: new IsolatedWorld(this), - [PUPPETEER_WORLD]: new IsolatedWorld(this), + [PUPPETEER_WORLD]: new IsolatedWorld(this, true), }; } diff --git a/src/common/IsolatedWorld.ts b/src/common/IsolatedWorld.ts index e969bdea..ff52fb95 100644 --- a/src/common/IsolatedWorld.ts +++ b/src/common/IsolatedWorld.ts @@ -15,11 +15,9 @@ */ import {Protocol} from 'devtools-protocol'; +import {source as injectedSource} from '../generated/injected.js'; import {assert} from '../util/assert.js'; -import { - createDeferredPromise, - DeferredPromise, -} from '../util/DeferredPromise.js'; +import {createDeferredPromise} from '../util/DeferredPromise.js'; import {CDPSession} from './Connection.js'; import {ElementHandle} from './ElementHandle.js'; import {TimeoutError} from './Errors.js'; @@ -117,8 +115,9 @@ export interface IsolatedWorldChart { */ export class IsolatedWorld { #frame: Frame; + #injected: boolean; #document?: ElementHandle; - #contextPromise: DeferredPromise = createDeferredPromise(); + #contextPromise = createDeferredPromise(); #detached = false; // Set of bindings that have been registered in the current context. @@ -140,10 +139,11 @@ export class IsolatedWorld { return `${name}_${contextId}`; }; - constructor(frame: Frame) { + constructor(frame: Frame, injected = false) { // Keep own reference to client because it might differ from the FrameManager's // client for OOP iframes. this.#frame = frame; + this.#injected = injected; this.#client.on('Runtime.bindingCalled', this.#onBindingCalled); } @@ -169,10 +169,9 @@ export class IsolatedWorld { } setContext(context: ExecutionContext): void { - assert( - this.#contextPromise, - `ExecutionContext ${context._contextId} has already been set.` - ); + if (this.#injected) { + context.evaluate(injectedSource).catch(debugError); + } this.#ctxBindings.clear(); this.#contextPromise.resolve(context); for (const waitTask of this._waitTasks) { diff --git a/src/injected/injected.ts b/src/injected/injected.ts index ed9e2f6b..f73435b3 100644 --- a/src/injected/injected.ts +++ b/src/injected/injected.ts @@ -1 +1,12 @@ -export * from './Poller.js'; +import * as Poller from './Poller.js'; +import * as util from './util.js'; + +Object.assign( + self, + Object.freeze({ + InjectedUtil: { + ...Poller, + ...util, + }, + }) +); diff --git a/src/injected/util.ts b/src/injected/util.ts new file mode 100644 index 00000000..79e68e5e --- /dev/null +++ b/src/injected/util.ts @@ -0,0 +1,18 @@ +const createdFunctions = new Map unknown>(); + +/** + * Creates a function from a string. + */ +export const createFunction = ( + functionValue: string +): ((...args: unknown[]) => unknown) => { + let fn = createdFunctions.get(functionValue); + if (fn) { + return fn; + } + fn = new Function(`return ${functionValue}`)() as ( + ...args: unknown[] + ) => unknown; + createdFunctions.set(functionValue, fn); + return fn; +}; diff --git a/src/templates/injected.ts.tmpl b/src/templates/injected.ts.tmpl new file mode 100644 index 00000000..377e59cb --- /dev/null +++ b/src/templates/injected.ts.tmpl @@ -0,0 +1,9 @@ +import * as Poller from '../injected/Poller.js'; +import * as util from '../injected/util.js'; + +declare global { + const InjectedUtil: Readonly; +} + +/** @internal */ +export const source = SOURCE_CODE; diff --git a/src/tsconfig.esm.json b/src/tsconfig.esm.json index f5444a18..aba2f240 100644 --- a/src/tsconfig.esm.json +++ b/src/tsconfig.esm.json @@ -8,5 +8,6 @@ "references": [ {"path": "../vendor/tsconfig.esm.json"}, {"path": "../compat/esm/tsconfig.json"} - ] + ], + "exclude": ["injected/injected.ts"] } diff --git a/test/src/injected.spec.ts b/test/src/injected.spec.ts new file mode 100644 index 00000000..b2fa5ed8 --- /dev/null +++ b/test/src/injected.spec.ts @@ -0,0 +1,40 @@ +/** + * Copyright 2022 Google Inc. All rights reserved. + * + * 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 expect from 'expect'; +import '../../lib/cjs/puppeteer/generated/injected.js'; +import {PUPPETEER_WORLD} from '../../lib/cjs/puppeteer/common/IsolatedWorld.js'; +import { + getTestState, + setupTestBrowserHooks, + setupTestPageAndContextHooks, +} from './mocha-utils.js'; + +describe('InjectedUtil tests', function () { + setupTestBrowserHooks(); + setupTestPageAndContextHooks(); + + it('should work', async () => { + const {page} = getTestState(); + + const handle = await page + .mainFrame() + .worlds[PUPPETEER_WORLD].evaluate(() => { + return typeof InjectedUtil === 'object'; + }); + expect(handle).toBeTruthy(); + }); +}); diff --git a/test/src/network.spec.ts b/test/src/network.spec.ts index 00e25585..9faad3bf 100644 --- a/test/src/network.spec.ts +++ b/test/src/network.spec.ts @@ -851,10 +851,13 @@ describe('network', function () { res.end(); }); await page.goto(httpsServer.PREFIX + '/setcookie.html'); - + const url = httpsServer.CROSS_PROCESS_PREFIX + '/setcookie.html'; const response = await new Promise(resolve => { - page.on('response', resolve); - const url = httpsServer.CROSS_PROCESS_PREFIX + '/setcookie.html'; + page.on('response', response => { + if (response.url() === url) { + resolve(response); + } + }); page.evaluate(src => { const xhr = new XMLHttpRequest(); xhr.open('GET', src); diff --git a/utils/generate_sources.ts b/utils/generate_sources.ts index a760bf9b..1dc919b8 100644 --- a/utils/generate_sources.ts +++ b/utils/generate_sources.ts @@ -17,27 +17,37 @@ import {job} from './internal/job.js'; .build(); await job('', async ({name, inputs, outputs}) => { + const input = inputs.find(input => { + return input.endsWith('injected.ts'); + })!; + const template = await readFile( + inputs.find(input => { + return input.includes('injected.ts.tmpl'); + })!, + 'utf8' + ); const tmp = await mkdtemp(name); await esbuild.build({ - entryPoints: [inputs[0]!], + entryPoints: [input], bundle: true, outdir: tmp, format: 'cjs', platform: 'browser', target: 'ES2019', }); - const baseName = path.basename(inputs[0]!); + const baseName = path.basename(input); const content = await readFile( path.join(tmp, baseName.replace('.ts', '.js')), 'utf-8' ); - const scriptContent = `/** @internal */ -export const source = ${JSON.stringify(content)}; -`; + const scriptContent = template.replace( + 'SOURCE_CODE', + JSON.stringify(content) + ); await writeFile(outputs[0]!, scriptContent); - await rimraf.sync(tmp); + rimraf.sync(tmp); }) - .inputs(['src/injected/**.ts']) + .inputs(['src/templates/injected.ts.tmpl', 'src/injected/**.ts']) .outputs(['src/generated/injected.ts']) .build(); diff --git a/utils/internal/job.ts b/utils/internal/job.ts index 5ccd2105..49226a50 100644 --- a/utils/internal/job.ts +++ b/utils/internal/job.ts @@ -59,8 +59,8 @@ class JobBuilder { ); if ( - outputStats.reduce(reduceMaxTime, 0) >= - inputStats.reduce(reduceMinTime, Infinity) + outputStats.reduce(reduceMinTime, Infinity) > + inputStats.reduce(reduceMaxTime, 0) ) { shouldRun = false; }