2025-03-21 10:58:58 -07:00
|
|
|
/**
|
|
|
|
* 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 * as playwright from 'playwright';
|
|
|
|
|
|
|
|
export class Context {
|
2025-03-26 15:02:45 -07:00
|
|
|
private _userDataDir: string;
|
2025-03-25 14:46:39 -07:00
|
|
|
private _launchOptions: playwright.LaunchOptions | undefined;
|
2025-03-25 13:05:28 -07:00
|
|
|
private _browser: playwright.Browser | undefined;
|
2025-03-21 10:58:58 -07:00
|
|
|
private _page: playwright.Page | undefined;
|
|
|
|
private _console: playwright.ConsoleMessage[] = [];
|
2025-03-26 15:02:45 -07:00
|
|
|
private _createPagePromise: Promise<playwright.Page> | undefined;
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-03-26 15:02:45 -07:00
|
|
|
constructor(userDataDir: string, launchOptions?: playwright.LaunchOptions) {
|
|
|
|
this._userDataDir = userDataDir;
|
2025-03-21 10:58:58 -07:00
|
|
|
this._launchOptions = launchOptions;
|
|
|
|
}
|
|
|
|
|
2025-03-26 15:02:45 -07:00
|
|
|
async createPage(): Promise<playwright.Page> {
|
|
|
|
if (this._createPagePromise)
|
|
|
|
return this._createPagePromise;
|
|
|
|
this._createPagePromise = (async () => {
|
|
|
|
const { browser, page } = await this._createPage();
|
|
|
|
page.on('console', event => this._console.push(event));
|
|
|
|
page.on('framenavigated', frame => {
|
2025-03-21 13:16:30 -07:00
|
|
|
if (!frame.parentFrame())
|
2025-03-21 13:19:51 -07:00
|
|
|
this._console.length = 0;
|
2025-03-21 13:16:30 -07:00
|
|
|
});
|
2025-03-26 15:02:45 -07:00
|
|
|
page.on('close', () => this._onPageClose());
|
|
|
|
this._page = page;
|
|
|
|
this._browser = browser;
|
|
|
|
return page;
|
2025-03-21 10:58:58 -07:00
|
|
|
})();
|
2025-03-26 15:02:45 -07:00
|
|
|
return this._createPagePromise;
|
2025-03-21 10:58:58 -07:00
|
|
|
}
|
|
|
|
|
2025-03-26 15:02:45 -07:00
|
|
|
private _onPageClose() {
|
2025-03-25 13:05:28 -07:00
|
|
|
const browser = this._browser;
|
2025-03-26 15:02:45 -07:00
|
|
|
const page = this._page;
|
|
|
|
void page?.context()?.close().then(() => browser?.close()).catch(() => {});
|
|
|
|
|
|
|
|
this._createPagePromise = undefined;
|
2025-03-25 13:05:28 -07:00
|
|
|
this._browser = undefined;
|
|
|
|
this._page = undefined;
|
|
|
|
this._console.length = 0;
|
|
|
|
}
|
|
|
|
|
2025-03-26 15:02:45 -07:00
|
|
|
async existingPage(): Promise<playwright.Page> {
|
|
|
|
if (!this._page)
|
|
|
|
throw new Error('Navigate to a location to create a page');
|
|
|
|
return this._page;
|
|
|
|
}
|
|
|
|
|
|
|
|
async console(): Promise<playwright.ConsoleMessage[]> {
|
|
|
|
return this._console;
|
|
|
|
}
|
|
|
|
|
|
|
|
async close() {
|
|
|
|
if (!this._page)
|
|
|
|
return;
|
|
|
|
await this._page.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
private async _createPage(): Promise<{ browser?: playwright.Browser, page: playwright.Page }> {
|
|
|
|
if (process.env.PLAYWRIGHT_WS_ENDPOINT) {
|
|
|
|
const url = new URL(process.env.PLAYWRIGHT_WS_ENDPOINT);
|
|
|
|
if (this._launchOptions)
|
|
|
|
url.searchParams.set('launch-options', JSON.stringify(this._launchOptions));
|
|
|
|
const browser = await playwright.chromium.connect(String(url));
|
|
|
|
const page = await browser.newPage();
|
|
|
|
return { browser, page };
|
|
|
|
}
|
|
|
|
|
|
|
|
const context = await playwright.chromium.launchPersistentContext(this._userDataDir, this._launchOptions);
|
|
|
|
const [page] = context.pages();
|
|
|
|
return { page };
|
2025-03-21 10:58:58 -07:00
|
|
|
}
|
|
|
|
}
|