/** * Copyright 2017 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 { helper } from './helper'; import * as network from './network'; import { Page, PageBinding } from './page'; import { TimeoutSettings } from './timeoutSettings'; import * as types from './types'; import { Events } from './events'; import { ExtendedEventEmitter } from './extendedEventEmitter'; import { Download } from './download'; export type BrowserContextOptions = { viewport?: types.Size | null, ignoreHTTPSErrors?: boolean, javaScriptEnabled?: boolean, bypassCSP?: boolean, userAgent?: string, locale?: string, timezoneId?: string, geolocation?: types.Geolocation, permissions?: string[], extraHTTPHeaders?: network.Headers, offline?: boolean, httpCredentials?: types.Credentials, deviceScaleFactor?: number, isMobile?: boolean, hasTouch?: boolean, colorScheme?: types.ColorScheme, acceptDownloads?: boolean }; export interface BrowserContext { setDefaultNavigationTimeout(timeout: number): void; setDefaultTimeout(timeout: number): void; pages(): Page[]; newPage(): Promise; cookies(urls?: string | string[]): Promise; addCookies(cookies: network.SetNetworkCookieParam[]): Promise; clearCookies(): Promise; grantPermissions(permissions: string[], options?: { origin?: string }): Promise; clearPermissions(): Promise; setGeolocation(geolocation: types.Geolocation | null): Promise; setExtraHTTPHeaders(headers: network.Headers): Promise; setOffline(offline: boolean): Promise; setHTTPCredentials(httpCredentials: types.Credentials | null): Promise; addInitScript(script: Function | string | { path?: string, content?: string }, arg?: any): Promise; exposeFunction(name: string, playwrightFunction: Function): Promise; route(url: types.URLMatch, handler: network.RouteHandler): Promise; waitForEvent(event: string, optionsOrPredicate?: Function | (types.TimeoutOptions & { predicate?: Function })): Promise; close(): Promise; } export abstract class BrowserContextBase extends ExtendedEventEmitter implements BrowserContext { readonly _timeoutSettings = new TimeoutSettings(); readonly _pageBindings = new Map(); readonly _options: BrowserContextOptions; readonly _routes: { url: types.URLMatch, handler: network.RouteHandler }[] = []; _closed = false; private readonly _closePromise: Promise; private _closePromiseFulfill: ((error: Error) => void) | undefined; readonly _permissions = new Map(); readonly _downloads = new Set(); constructor(options: BrowserContextOptions) { super(); this._options = options; this._closePromise = new Promise(fulfill => this._closePromiseFulfill = fulfill); } protected _abortPromiseForEvent(event: string) { return event === Events.BrowserContext.Close ? super._abortPromiseForEvent(event) : this._closePromise; } protected _timeoutForEvent(event: string): number { return this._timeoutSettings.timeout(); } _browserClosed() { for (const page of this.pages()) page._didClose(); this._didCloseInternal(true); } async _didCloseInternal(omitDeleteDownloads = false) { this._closed = true; this.emit(Events.BrowserContext.Close); this._closePromiseFulfill!(new Error('Context closed')); if (!omitDeleteDownloads) await Promise.all([...this._downloads].map(d => d.delete())); this._downloads.clear(); } // BrowserContext methods. abstract pages(): Page[]; abstract newPage(): Promise; abstract cookies(...urls: string[]): Promise; abstract addCookies(cookies: network.SetNetworkCookieParam[]): Promise; abstract clearCookies(): Promise; abstract _doGrantPermissions(origin: string, permissions: string[]): Promise; abstract _doClearPermissions(): Promise; abstract setGeolocation(geolocation: types.Geolocation | null): Promise; abstract setHTTPCredentials(httpCredentials: types.Credentials | null): Promise; abstract setExtraHTTPHeaders(headers: network.Headers): Promise; abstract setOffline(offline: boolean): Promise; abstract addInitScript(script: string | Function | { path?: string | undefined; content?: string | undefined; }, arg?: any): Promise; abstract exposeFunction(name: string, playwrightFunction: Function): Promise; abstract route(url: types.URLMatch, handler: network.RouteHandler): Promise; abstract close(): Promise; async grantPermissions(permissions: string[], options?: { origin?: string }) { let origin = '*'; if (options && options.origin) { const url = new URL(options.origin); origin = url.origin; } const existing = new Set(this._permissions.get(origin) || []); permissions.forEach(p => existing.add(p)); const list = [...existing.values()]; this._permissions.set(origin, list); await this._doGrantPermissions(origin, list); } async clearPermissions() { this._permissions.clear(); await this._doClearPermissions(); } setDefaultNavigationTimeout(timeout: number) { this._timeoutSettings.setDefaultNavigationTimeout(timeout); } setDefaultTimeout(timeout: number) { this._timeoutSettings.setDefaultTimeout(timeout); } } export function assertBrowserContextIsNotOwned(context: BrowserContextBase) { for (const page of context.pages()) { if (page._ownedContext) throw new Error('Please use browser.newContext() for multi-page scripts that share the context.'); } } export function validateBrowserContextOptions(options: BrowserContextOptions): BrowserContextOptions { const result = { ...options }; if (!result.viewport && result.viewport !== null) result.viewport = { width: 1280, height: 720 }; if (result.viewport) result.viewport = { ...result.viewport }; if (result.geolocation) result.geolocation = verifyGeolocation(result.geolocation); if (result.extraHTTPHeaders) result.extraHTTPHeaders = network.verifyHeaders(result.extraHTTPHeaders); return result; } export function verifyGeolocation(geolocation: types.Geolocation): types.Geolocation { const result = { ...geolocation }; result.accuracy = result.accuracy || 0; const { longitude, latitude, accuracy } = result; if (!helper.isNumber(longitude) || longitude < -180 || longitude > 180) throw new Error(`Invalid longitude "${longitude}": precondition -180 <= LONGITUDE <= 180 failed.`); if (!helper.isNumber(latitude) || latitude < -90 || latitude > 90) throw new Error(`Invalid latitude "${latitude}": precondition -90 <= LATITUDE <= 90 failed.`); if (!helper.isNumber(accuracy) || accuracy < 0) throw new Error(`Invalid accuracy "${accuracy}": precondition 0 <= ACCURACY failed.`); return result; }