From 2a0eefb99f0ae00dacc9e768a253308c0d18a4c3 Mon Sep 17 00:00:00 2001 From: Alex Rudenko Date: Wed, 17 Aug 2022 19:34:34 +0200 Subject: [PATCH] fix: handle service workers in extensions (#8807) Closes #8800 --- src/common/ChromeTargetManager.ts | 9 +++--- .../serviceworkers/extension/background.js | 1 + .../serviceworkers/extension/manifest.json | 9 ++++++ test/src/headful.spec.ts | 31 +++++++++++++++++-- 4 files changed, 44 insertions(+), 6 deletions(-) create mode 100644 test/assets/serviceworkers/extension/background.js create mode 100644 test/assets/serviceworkers/extension/manifest.json diff --git a/src/common/ChromeTargetManager.ts b/src/common/ChromeTargetManager.ts index f08001e0..445225ed 100644 --- a/src/common/ChromeTargetManager.ts +++ b/src/common/ChromeTargetManager.ts @@ -317,11 +317,12 @@ export class ChromeTargetManager extends EventEmitter implements TargetManager { ) { this.#finishInitializationIfReady(targetInfo.targetId); await silentDetach(); - if (parentSession instanceof CDPSession) { - const target = this.#targetFactory(targetInfo); - this.#attachedTargetsByTargetId.set(targetInfo.targetId, target); - this.emit(TargetManagerEmittedEvents.TargetAvailable, target); + if (this.#attachedTargetsByTargetId.has(targetInfo.targetId)) { + return; } + const target = this.#targetFactory(targetInfo); + this.#attachedTargetsByTargetId.set(targetInfo.targetId, target); + this.emit(TargetManagerEmittedEvents.TargetAvailable, target); return; } diff --git a/test/assets/serviceworkers/extension/background.js b/test/assets/serviceworkers/extension/background.js new file mode 100644 index 00000000..8b1a3937 --- /dev/null +++ b/test/assets/serviceworkers/extension/background.js @@ -0,0 +1 @@ +// empty diff --git a/test/assets/serviceworkers/extension/manifest.json b/test/assets/serviceworkers/extension/manifest.json new file mode 100644 index 00000000..25828b6d --- /dev/null +++ b/test/assets/serviceworkers/extension/manifest.json @@ -0,0 +1,9 @@ +{ + "name": "Simple extension", + "version": "0.1", + "background": { + "service_worker": "background.js" + }, + "permissions": ["background", "activeTab"], + "manifest_version": 3 +} diff --git a/test/src/headful.spec.ts b/test/src/headful.spec.ts index 1e546978..95abfaa2 100644 --- a/test/src/headful.spec.ts +++ b/test/src/headful.spec.ts @@ -36,6 +36,13 @@ const mkdtempAsync = promisify(fs.mkdtemp); const TMP_FOLDER = path.join(os.tmpdir(), 'pptr_tmp_folder-'); const extensionPath = path.join(__dirname, '../assets', 'simple-extension'); +const serviceWorkerExtensionPath = path.join( + __dirname, + '..', + 'assets', + 'serviceworkers', + 'extension' +); describeChromeOnly('headful tests', function () { /* These tests fire up an actual browser so let's @@ -120,7 +127,7 @@ describeChromeOnly('headful tests', function () { ); const page = await browserWithExtension.newPage(); const backgroundPageTarget = await browserWithExtension.waitForTarget( - (target: {type: () => string}) => { + target => { return target.type() === 'background_page'; } ); @@ -128,6 +135,26 @@ describeChromeOnly('headful tests', function () { await browserWithExtension.close(); expect(backgroundPageTarget).toBeTruthy(); }); + it('service_worker target type should be available', async () => { + const {puppeteer, defaultBrowserOptions} = getTestState(); + const browserWithExtension = await launchBrowser(puppeteer, { + ...defaultBrowserOptions, + headless: false, + args: [ + `--disable-extensions-except=${serviceWorkerExtensionPath}`, + `--load-extension=${serviceWorkerExtensionPath}`, + ], + }); + const page = await browserWithExtension.newPage(); + const serviceWorkerTarget = await browserWithExtension.waitForTarget( + target => { + return target.type() === 'service_worker'; + } + ); + await page.close(); + await browserWithExtension.close(); + expect(serviceWorkerTarget).toBeTruthy(); + }); it('target.page() should return a background_page', async function () { const {puppeteer} = getTestState(); const browserWithExtension = await launchBrowser( @@ -135,7 +162,7 @@ describeChromeOnly('headful tests', function () { extensionOptions ); const backgroundPageTarget = await browserWithExtension.waitForTarget( - (target: {type: () => string}) => { + target => { return target.type() === 'background_page'; } );