/**
* @license
* Copyright 2018 Google Inc.
* SPDX-License-Identifier: Apache-2.0
*/
import type {ServerResponse} from 'http' ;
import expect from 'expect' ;
import {type Target, TimeoutError} from 'puppeteer' ;
import type {Page} from 'puppeteer-core/internal/api/Page.js' ;
import {getTestState, setupTestBrowserHooks} from './mocha-utils.js' ;
import {waitEvent} from './utils.js' ;
describe('Target' , function () {
setupTestBrowserHooks();
it('Browser.targets should return all of the targets' , async () => {
const {browser} = await getTestState();
// The pages will be the testing page and the original newtab page
const targets = browser.targets();
expect(
targets.some(target => {
return target.type() === 'page' && target.url() === 'about:blank' ;
}),
).toBeTruthy();
expect(
targets.some(target => {
return target.type() === 'browser' ;
}),
).toBeTruthy();
});
it('Browser.pages should return all of the pages' , async () => {
const {page, context} = await getTestState();
// The pages will be the testing page
const allPages = await context.pages();
expect(allPages).toHaveLength(1 );
expect(allPages).toContain(page);
});
it('should contain browser target' , async () => {
const {browser} = await getTestState();
const targets = browser.targets();
const browserTarget = targets.find(target => {
return target.type() === 'browser' ;
});
expect(browserTarget).toBeTruthy();
});
it('should be able to use the default page in the browser' , async () => {
const {page, browser} = await getTestState();
// The pages will be the testing page and the original newtab page
const allPages = await browser.pages();
const originalPage = allPages.find(p => {
return p !== page;
})!;
expect(
await originalPage.evaluate(() => {
return ['Hello' , 'world' ].join(' ' );
}),
).toBe('Hello world' );
expect(await originalPage.$('body' )).toBeTruthy();
});
it('should be able to use async waitForTarget' , async () => {
const {page, server, context} = await getTestState();
const [otherPage] = await Promise.all([
context
.waitForTarget(
target => {
return target.page().then(page => {
return (
page!.url() === server.CROSS_PROCESS_PREFIX + '/empty.html'
);
});
},
{timeout: 3000 },
)
.then(target => {
return target.page();
}),
page.evaluate((url: string) => {
return window.open(url);
}, server.CROSS_PROCESS_PREFIX + '/empty.html' ),
]);
expect(otherPage!.url()).toEqual(
server.CROSS_PROCESS_PREFIX + '/empty.html' ,
);
expect(page).not.toBe(otherPage);
});
it('should report when a new page is created and closed' , async () => {
const {page, server, context} = await getTestState();
const [otherPage] = await Promise.all([
context
.waitForTarget(
target => {
return target.url() === server.CROSS_PROCESS_PREFIX + '/empty.html' ;
},
{timeout: 3000 },
)
.then(target => {
return target.page();
}),
page.evaluate((url: string) => {
return window.open(url);
}, server.CROSS_PROCESS_PREFIX + '/empty.html' ),
]);
expect(otherPage!.url()).toContain(server.CROSS_PROCESS_PREFIX);
expect(
await otherPage!.evaluate(() => {
return ['Hello' , 'world' ].join(' ' );
}),
).toBe('Hello world' );
expect(await otherPage!.$('body' )).toBeTruthy();
let allPages = await context.pages();
expect(allPages).toContain(page);
expect(allPages).toContain(otherPage);
const [closedTarget] = await Promise.all([
waitEvent<Target>(context, 'targetdestroyed' ),
otherPage!.close(),
]);
expect(await closedTarget.page()).toBe(otherPage);
allPages = (await Promise.all(
context.targets().map(target => {
return target.page();
}),
)) as Page[];
expect(allPages).toContain(page);
expect(allPages).not.toContain(otherPage);
});
it('should report when a service worker is created and destroyed' , async () => {
const {page, server, context} = await getTestState();
await page.goto (server.EMPTY_PAGE);
const createdTarget = waitEvent(context, 'targetcreated' );
await page.goto (server.PREFIX + '/serviceworkers/empty/sw.html' );
expect((await createdTarget).type()).toBe('service_worker' );
expect((await createdTarget).url()).toBe(
server.PREFIX + '/serviceworkers/empty/sw.js' ,
);
const destroyedTarget = waitEvent(context, 'targetdestroyed' );
await page.evaluate(() => {
return (
globalThis as unknown as {
registrationPromise: Promise<{unregister: () => void }>;
}
).registrationPromise.then((registration: any) => {
return registration.unregister();
});
});
expect(await destroyedTarget).toBe(await createdTarget);
});
it('should create a worker from a service worker' , async () => {
const {page, server, context} = await getTestState();
await page.goto (server.PREFIX + '/serviceworkers/empty/sw.html' );
const target = await context.waitForTarget(
target => {
return target.type() === 'service_worker' ;
},
{timeout: 3000 },
);
const worker = (await target.worker())!;
expect(
await worker.evaluate(() => {
return self.toString();
}),
).toBe('[object ServiceWorkerGlobalScope]' );
});
it('should close a service worker' , async () => {
const {page, server, context} = await getTestState();
await page.goto (server.PREFIX + '/serviceworkers/empty/sw.html' );
const target = await context.waitForTarget(
target => {
return target.type() === 'service_worker' ;
},
{timeout: 3000 },
);
const worker = (await target.worker())!;
const onceDestroyed = new Promise(resolve => {
context.once('targetdestroyed' , event => {
resolve(event);
});
});
await worker.close();
expect(await onceDestroyed).toBe(target);
});
it('should create a worker from a shared worker' , async () => {
const {page, server, context} = await getTestState();
await page.goto (server.EMPTY_PAGE);
await page.evaluate(() => {
new SharedWorker('data:text/javascript,console.log("hi")' );
});
const target = await context.waitForTarget(
target => {
return target.type() === 'shared_worker' ;
},
{timeout: 3000 },
);
const worker = (await target.worker())!;
expect(
await worker.evaluate(() => {
return self.toString();
}),
).toBe('[object SharedWorkerGlobalScope]' );
});
it('should close a shared worker' , async () => {
const {page, server, context} = await getTestState();
await page.goto (server.EMPTY_PAGE);
await page.evaluate(() => {
new SharedWorker('data:text/javascript,console.log("hi2")' );
});
const target = await context.waitForTarget(
target => {
return target.type() === 'shared_worker' ;
},
{timeout: 3000 },
);
const worker = (await target.worker())!;
const onceDestroyed = new Promise(resolve => {
context.once('targetdestroyed' , event => {
resolve(event);
});
});
await worker.close();
expect(await onceDestroyed).toBe(target);
});
it('should report when a target url changes' , async () => {
const {page, server, context} = await getTestState();
await page.goto (server.EMPTY_PAGE);
let changedTarget = waitEvent(context, 'targetchanged' );
await page.goto (server.CROSS_PROCESS_PREFIX + '/' );
expect((await changedTarget).url()).toBe(server.CROSS_PROCESS_PREFIX + '/' );
changedTarget = waitEvent(context, 'targetchanged' );
await page.goto (server.EMPTY_PAGE);
expect((await changedTarget).url()).toBe(server.EMPTY_PAGE);
});
it('should not report uninitialized pages' , async () => {
const {context} = await getTestState();
let targetChanged = false ;
const listener = () => {
targetChanged = true ;
};
context.on('targetchanged' , listener);
const targetPromise = waitEvent<Target>(context, 'targetcreated' );
const newPagePromise = context.newPage();
const target = await targetPromise;
expect(target.url()).toBe('about:blank' );
const newPage = await newPagePromise;
const targetPromise2 = waitEvent<Target>(context, 'targetcreated' );
const evaluatePromise = newPage.evaluate(() => {
return window.open('about:blank' );
});
const target2 = await targetPromise2;
expect(target2.url()).toBe('about:blank' );
await evaluatePromise;
await newPage.close();
expect(targetChanged).toBe(false );
context.off('targetchanged' , listener);
});
it('should not crash while redirecting if original request was missed' , async () => {
const {page, server, context} = await getTestState();
let serverResponse!: ServerResponse;
server.setRoute('/one-style.css' , (_req, res) => {
return (serverResponse = res);
});
// Open a new page. Use window.open to connect to the page later.
await Promise.all([
page.evaluate((url: string) => {
return window.open(url);
}, server.PREFIX + '/one-style.html' ),
server.waitForRequest('/one-style.css' ),
]);
// Connect to the opened page.
const target = await context.waitForTarget(
target => {
return target.url().includes('one-style.html' );
},
{timeout: 3000 },
);
const newPage = (await target.page())!;
const loadEvent = waitEvent(newPage, 'load' );
// Issue a redirect.
serverResponse.writeHead(302 , {location: '/injectedstyle.css' });
serverResponse.end();
// Wait for the new page to load.
await loadEvent;
// Cleanup.
await newPage.close();
});
it('should have an opener' , async () => {
const {page, server, context} = await getTestState();
await page.goto (server.EMPTY_PAGE);
const [createdTarget] = await Promise.all([
waitEvent<Target>(context, 'targetcreated' ),
page.goto (server.PREFIX + '/popup/window-open.html' ),
]);
expect((await createdTarget.page())!.url()).toBe(
server.PREFIX + '/popup/popup.html' ,
);
expect(createdTarget.opener()).toBe(page.target());
expect(page.target().opener()).toBeUndefined();
});
describe('Browser.waitForTarget' , () => {
it('should wait for a target' , async () => {
const {browser, server, context} = await getTestState();
let resolved = false ;
const targetPromise = browser.waitForTarget(
target => {
return target.url() === server.EMPTY_PAGE;
},
{timeout: 3000 },
);
targetPromise
.then(() => {
return (resolved = true );
})
.catch (error => {
resolved = true ;
if (error instanceof TimeoutError) {
console.error(error);
} else {
throw error;
}
});
const page = await context.newPage();
expect(resolved).toBe(false );
await page.goto (server.EMPTY_PAGE);
try {
const target = await targetPromise;
expect(await target.page()).toBe(page);
} catch (error) {
if (error instanceof TimeoutError) {
console.error(error);
} else {
throw error;
}
}
await page.close();
});
it('should timeout waiting for a non-existent target' , async () => {
const {browser, server} = await getTestState();
let error!: Error;
await browser
.waitForTarget(
target => {
return target.url() === server.PREFIX + '/does-not-exist.html' ;
},
{
timeout: 1 ,
},
)
.catch (error_ => {
return (error = error_);
});
expect(error).toBeInstanceOf(TimeoutError);
});
it('should be able to abort' , async () => {
const {browser} = await getTestState();
const abortController = new AbortController();
const task = browser.waitForTarget(
() => {
return false ;
},
{
signal: abortController.signal,
},
);
abortController.abort();
await expect(task).rejects.toThrow(/aborted/);
});
});
});
Messung V0.5 in Prozent C=97 H=88 G=92
¤ Dauer der Verarbeitung: 0.27 Sekunden
(vorverarbeitet am 2026-06-04)
¤
*© Formatika GbR, Deutschland