TenantAtlas/apps/platform/.pnpm-store/v10/files/fb/65b66f97284c6a6571aab30ec2689323cdcb87c4e065e155f96db46b6bd8cc1699e3dc1e18156bc49d9108f23aa6ad6feeef768604a8397ad7bcad77a081e9
ahmido 1fec9c6f9d
Some checks failed
Main Confidence / confidence (push) Failing after 45s
feat: compress governance operator outcomes (#253)
## Summary
- introduce surface-aware compressed governance outcomes and reuse the shared truth/explanation seams for operator-first summaries
- apply the compressed outcome hierarchy across baseline, evidence, review, review-pack, canonical review/evidence, and artifact-oriented operation-run surfaces
- expand spec 214 fixtures and Pest coverage, and fix tenant-panel route assertions by generating explicit tenant-panel URLs in the affected Filament tests

## Validation
- `cd apps/platform && ./vendor/bin/sail bin pint --dirty --format agent`
- focused governance compression suite from `specs/214-governance-outcome-compression/quickstart.md` passed (`68` tests, `445` assertions)
- `cd apps/platform && ./vendor/bin/sail artisan test --compact tests/Feature/Filament/InventoryItemResourceTest.php tests/Feature/Filament/BackupSetUiEnforcementTest.php tests/Feature/Filament/RestoreRunUiEnforcementTest.php` passed (`18` tests, `81` assertions)

Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de>
Reviewed-on: #253
2026-04-19 12:30:36 +00:00

742 lines
28 KiB
Plaintext

"use strict";
var __create = Object.create;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __getProtoOf = Object.getPrototypeOf;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
// If the importer is in node compatibility mode or this is not an ESM
// file that has been converted to a CommonJS file using a Babel-
// compatible transform (i.e. "__esModule" has not been set), then set
// "default" to the CommonJS "module.exports" for node compatibility.
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
mod
));
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var browserContext_exports = {};
__export(browserContext_exports, {
BrowserContext: () => BrowserContext,
calculateUserAgentEmulation: () => calculateUserAgentEmulation,
normalizeProxySettings: () => normalizeProxySettings,
validateBrowserContextOptions: () => validateBrowserContextOptions,
verifyClientCertificates: () => verifyClientCertificates,
verifyGeolocation: () => verifyGeolocation
});
module.exports = __toCommonJS(browserContext_exports);
var import_fs = __toESM(require("fs"));
var import_crypto = require("./utils/crypto");
var import_debug = require("./utils/debug");
var import_clock = require("./clock");
var import_debugger = require("./debugger");
var import_dialog = require("./dialog");
var import_fetch = require("./fetch");
var import_stackTrace = require("../utils/isomorphic/stackTrace");
var import_harRecorder = require("./har/harRecorder");
var import_helper = require("./helper");
var import_instrumentation = require("./instrumentation");
var network = __toESM(require("./network"));
var import_page = require("./page");
var import_page2 = require("./page");
var import_recorderApp = require("./recorder/recorderApp");
var import_selectors = require("./selectors");
var import_tracing = require("./trace/recorder/tracing");
var rawStorageSource = __toESM(require("../generated/storageScriptSource"));
const BrowserContextEvent = {
Console: "console",
Close: "close",
Page: "page",
// Can't use just 'error' due to node.js special treatment of error events.
// @see https://nodejs.org/api/events.html#events_error_events
PageError: "pageerror",
Request: "request",
Response: "response",
RequestFailed: "requestfailed",
RequestFinished: "requestfinished",
RequestAborted: "requestaborted",
RequestFulfilled: "requestfulfilled",
RequestContinued: "requestcontinued",
BeforeClose: "beforeclose",
RecorderEvent: "recorderevent",
PageClosed: "pageclosed",
InternalFrameNavigatedToNewDocument: "internalframenavigatedtonewdocument"
};
class BrowserContext extends import_instrumentation.SdkObject {
constructor(browser, options, browserContextId) {
super(browser, "browser-context");
this._pageBindings = /* @__PURE__ */ new Map();
this.requestInterceptors = [];
this._closedStatus = "open";
this._permissions = /* @__PURE__ */ new Map();
this._downloads = /* @__PURE__ */ new Set();
this._origins = /* @__PURE__ */ new Set();
this._harRecorders = /* @__PURE__ */ new Map();
this._tempDirs = [];
this._creatingStorageStatePage = false;
this.initScripts = [];
this._routesInFlight = /* @__PURE__ */ new Set();
this._consoleApiExposed = false;
this.attribution.context = this;
this._browser = browser;
this._options = options;
this._browserContextId = browserContextId;
this._isPersistentContext = !browserContextId;
this._closePromise = new Promise((fulfill) => this._closePromiseFulfill = fulfill);
this._selectors = new import_selectors.Selectors(options.selectorEngines || [], options.testIdAttributeName);
this.fetchRequest = new import_fetch.BrowserContextAPIRequestContext(this);
this.tracing = new import_tracing.Tracing(this, browser.options.tracesDir);
this.clock = new import_clock.Clock(this);
this.dialogManager = new import_dialog.DialogManager(this.instrumentation);
}
static {
this.Events = BrowserContextEvent;
}
isPersistentContext() {
return this._isPersistentContext;
}
selectors() {
return this._selectors;
}
async _initialize() {
if (this.attribution.playwright.options.isInternalPlaywright)
return;
this._debugger = new import_debugger.Debugger(this);
const shouldEnableDebugger = !this.attribution.playwright.options.isServer && ((0, import_debug.isUnderTest)() || !!this._browser.options.headful);
if (shouldEnableDebugger) {
this._debugger.setPauseAt();
this._debugger.on(import_debugger.Debugger.Events.PausedStateChanged, () => {
if (this._debugger.isPaused())
import_recorderApp.RecorderApp.showInspectorNoReply(this);
});
}
if ((0, import_debug.debugMode)() === "inspector") {
this._debugger.setPauseAt({ next: true });
await import_recorderApp.RecorderApp.show(this, { pauseOnNextStatement: true });
}
if ((0, import_debug.debugMode)() === "console")
await this.exposeConsoleApi();
if (this._options.serviceWorkers === "block")
await this.addInitScript(`
if (navigator.serviceWorker) navigator.serviceWorker.register = async () => { console.warn('Service Worker registration blocked by Playwright'); };
`);
if (this._options.permissions)
await this.grantPermissions(this._options.permissions);
}
debugger() {
return this._debugger;
}
async exposeConsoleApi() {
if (this._consoleApiExposed)
return;
this._consoleApiExposed = true;
await this.extendInjectedScript(`
function installConsoleApi(injectedScript) { injectedScript.consoleApi.install(); }
module.exports = { default: () => installConsoleApi };
`);
}
canResetForReuse() {
if (this._closedStatus !== "open")
return false;
return true;
}
static reusableContextHash(params) {
const paramsCopy = { ...params };
if (paramsCopy.selectorEngines?.length === 0)
delete paramsCopy.selectorEngines;
for (const k of Object.keys(paramsCopy)) {
const key = k;
if (paramsCopy[key] === defaultNewContextParamValues[key])
delete paramsCopy[key];
}
for (const key of paramsThatAllowContextReuse)
delete paramsCopy[key];
return JSON.stringify(paramsCopy);
}
async resetForReuse(progress, params) {
await this.tracing.resetForReuse(progress);
if (params) {
for (const key of paramsThatAllowContextReuse)
this._options[key] = params[key];
if (params.testIdAttributeName)
this.selectors().setTestIdAttributeName(params.testIdAttributeName);
}
let page = this.pages()[0];
const otherPages = this.possiblyUninitializedPages().filter((p) => p !== page);
for (const p of otherPages)
await p.close();
if (page && page.hasCrashed()) {
await page.close();
page = void 0;
}
await page?.mainFrame().gotoImpl(progress, "about:blank", {});
await this.clock.uninstall(progress);
await progress.race(this.setUserAgent(this._options.userAgent));
await progress.race(this.doUpdateDefaultEmulatedMedia());
await progress.race(this.doUpdateDefaultViewport());
await this.setStorageState(progress, this._options.storageState, "resetForReuse");
await page?.resetForReuse(progress);
}
_browserClosed() {
for (const page of this.pages())
page._didClose();
this._didCloseInternal();
}
_didCloseInternal() {
if (this._closedStatus === "closed") {
return;
}
this._clientCertificatesProxy?.close().catch(() => {
});
this.tracing.abort();
if (this._isPersistentContext)
this.onClosePersistent();
this._closePromiseFulfill(new Error("Context closed"));
this.emit(BrowserContext.Events.Close);
}
pages() {
return this.possiblyUninitializedPages().filter((page) => page.initializedOrUndefined());
}
async cookies(urls = []) {
if (urls && !Array.isArray(urls))
urls = [urls];
return await this.doGetCookies(urls);
}
async clearCookies(options) {
const currentCookies = await this.cookies();
await this.doClearCookies();
const matches = (cookie, prop, value) => {
if (!value)
return true;
if (value instanceof RegExp) {
value.lastIndex = 0;
return value.test(cookie[prop]);
}
return cookie[prop] === value;
};
const cookiesToReadd = currentCookies.filter((cookie) => {
return !matches(cookie, "name", options.name) || !matches(cookie, "domain", options.domain) || !matches(cookie, "path", options.path);
});
await this.addCookies(cookiesToReadd);
}
setHTTPCredentials(httpCredentials) {
return this.doSetHTTPCredentials(httpCredentials);
}
getBindingClient(name) {
return this._pageBindings.get(name)?.forClient;
}
async exposePlaywrightBindingIfNeeded() {
this._playwrightBindingExposed ??= (async () => {
await this.doExposePlaywrightBinding();
this.bindingsInitScript = import_page2.PageBinding.createInitScript(this);
this.initScripts.push(this.bindingsInitScript);
await this.doAddInitScript(this.bindingsInitScript);
await this.safeNonStallingEvaluateInAllFrames(this.bindingsInitScript.source, "main");
})();
return await this._playwrightBindingExposed;
}
needsPlaywrightBinding() {
return this._playwrightBindingExposed !== void 0;
}
async exposeBinding(progress, name, needsHandle, playwrightBinding, forClient) {
if (this._pageBindings.has(name))
throw new Error(`Function "${name}" has been already registered`);
for (const page of this.pages()) {
if (page.getBinding(name))
throw new Error(`Function "${name}" has been already registered in one of the pages`);
}
await progress.race(this.exposePlaywrightBindingIfNeeded());
const binding = new import_page2.PageBinding(this, name, playwrightBinding, needsHandle);
binding.forClient = forClient;
this._pageBindings.set(name, binding);
try {
await progress.race(this.doAddInitScript(binding.initScript));
await progress.race(this.safeNonStallingEvaluateInAllFrames(binding.initScript.source, "main"));
return binding;
} catch (error) {
this._pageBindings.delete(name);
throw error;
}
}
async removeExposedBinding(binding) {
if (this._pageBindings.get(binding.name) !== binding)
return;
this._pageBindings.delete(binding.name);
await this.doRemoveInitScripts([binding.initScript]);
const cleanup = `{ ${binding.cleanupScript} };`;
await this.safeNonStallingEvaluateInAllFrames(cleanup, "main");
}
async grantPermissions(permissions, origin) {
let resolvedOrigin = "*";
if (origin) {
const url = new URL(origin);
resolvedOrigin = url.origin;
}
const existing = new Set(this._permissions.get(resolvedOrigin) || []);
permissions.forEach((p) => existing.add(p));
const list = [...existing.values()];
this._permissions.set(resolvedOrigin, list);
await this.doGrantPermissions(resolvedOrigin, list);
}
async clearPermissions() {
this._permissions.clear();
await this.doClearPermissions();
}
async setExtraHTTPHeaders(progress, headers) {
const oldHeaders = this._options.extraHTTPHeaders;
this._options.extraHTTPHeaders = headers;
try {
await progress.race(this.doUpdateExtraHTTPHeaders());
} catch (error) {
this._options.extraHTTPHeaders = oldHeaders;
this.doUpdateExtraHTTPHeaders().catch(() => {
});
throw error;
}
}
async setOffline(progress, offline) {
const oldOffline = this._options.offline;
this._options.offline = offline;
try {
await progress.race(this.doUpdateOffline());
} catch (error) {
this._options.offline = oldOffline;
this.doUpdateOffline().catch(() => {
});
throw error;
}
}
async _loadDefaultContextAsIs(progress) {
if (!this.possiblyUninitializedPages().length) {
const waitForEvent = import_helper.helper.waitForEvent(progress, this, BrowserContext.Events.Page);
await Promise.race([waitForEvent.promise, this._closePromise]);
}
const page = this.possiblyUninitializedPages()[0];
if (!page)
return;
const pageOrError = await progress.race(page.waitForInitializedOrError());
if (pageOrError instanceof Error)
throw pageOrError;
await page.mainFrame().waitForLoadState(progress, "load");
return page;
}
async _loadDefaultContext(progress) {
const defaultPage = await this._loadDefaultContextAsIs(progress);
if (!defaultPage)
return;
const browserName = this._browser.options.name;
if (this._options.isMobile && browserName === "chromium" || this._options.locale && browserName === "webkit") {
await this.newPage(progress);
await defaultPage.close();
}
}
_authenticateProxyViaHeader() {
const proxy = this._options.proxy || this._browser.options.proxy || { username: void 0, password: void 0 };
const { username, password } = proxy;
if (username) {
this._options.httpCredentials = { username, password };
const token = Buffer.from(`${username}:${password}`).toString("base64");
this._options.extraHTTPHeaders = network.mergeHeaders([
this._options.extraHTTPHeaders,
network.singleHeader("Proxy-Authorization", `Basic ${token}`)
]);
}
}
_authenticateProxyViaCredentials() {
const proxy = this._options.proxy || this._browser.options.proxy;
if (!proxy)
return;
const { username, password } = proxy;
if (username)
this._options.httpCredentials = { username, password: password || "" };
}
async addInitScript(source) {
const initScript = new import_page.InitScript(this, source);
this.initScripts.push(initScript);
try {
await this.doAddInitScript(initScript);
return initScript;
} catch (error) {
initScript.dispose().catch(() => {
});
throw error;
}
}
async removeInitScript(initScript) {
this.initScripts = this.initScripts.filter((script) => initScript !== script);
await this.doRemoveInitScripts([initScript]);
}
async addRequestInterceptor(progress, handler) {
this.requestInterceptors.push(handler);
await this.doUpdateRequestInterception();
}
async removeRequestInterceptor(handler) {
const index = this.requestInterceptors.indexOf(handler);
if (index === -1)
return;
this.requestInterceptors.splice(index, 1);
await this.notifyRoutesInFlightAboutRemovedHandler(handler);
await this.doUpdateRequestInterception();
}
isClosingOrClosed() {
return this._closedStatus !== "open";
}
async _deleteAllDownloads() {
await Promise.all(Array.from(this._downloads).map((download) => download.artifact.deleteOnContextClose()));
}
async _deleteAllTempDirs() {
await Promise.all(this._tempDirs.map(async (dir) => await import_fs.default.promises.unlink(dir).catch((e) => {
})));
}
setCustomCloseHandler(handler) {
this._customCloseHandler = handler;
}
async close(options) {
if (this._closedStatus === "open") {
if (options.reason)
this._closeReason = options.reason;
this.emit(BrowserContext.Events.BeforeClose);
this._closedStatus = "closing";
for (const harRecorder of this._harRecorders.values())
await harRecorder.flush();
await this.tracing.flush();
await Promise.all(this.pages().map((page) => page.screencast.handlePageOrContextClose()));
if (this._customCloseHandler) {
await this._customCloseHandler();
} else {
await this.doClose(options.reason);
}
const promises = [];
promises.push(this._deleteAllDownloads());
promises.push(this._deleteAllTempDirs());
await Promise.all(promises);
if (!this._customCloseHandler)
this._didCloseInternal();
}
await this._closePromise;
}
async newPage(progress, forStorageState) {
let page;
try {
this._creatingStorageStatePage = !!forStorageState;
page = await progress.race(this.doCreateNewPage());
const pageOrError = await progress.race(page.waitForInitializedOrError());
if (pageOrError instanceof import_page2.Page) {
if (pageOrError.isClosed())
throw new Error("Page has been closed.");
return pageOrError;
}
throw pageOrError;
} catch (error) {
await page?.close({ reason: "Failed to create page" }).catch(() => {
});
throw error;
} finally {
this._creatingStorageStatePage = false;
}
}
addVisitedOrigin(origin) {
this._origins.add(origin);
}
async storageState(progress, indexedDB = false) {
const result = {
cookies: await this.cookies(),
origins: []
};
const originsToSave = new Set(this._origins);
const collectScript = `(() => {
const module = {};
${rawStorageSource.source}
const script = new (module.exports.StorageScript())(${this._browser.options.name === "firefox"});
return script.collect(${indexedDB});
})()`;
for (const page of this.pages()) {
const origin = page.mainFrame().origin();
if (!origin || !originsToSave.has(origin))
continue;
try {
const storage = await page.mainFrame().nonStallingEvaluateInExistingContext(collectScript, "utility");
if (storage.localStorage.length || storage.indexedDB?.length)
result.origins.push({ origin, localStorage: storage.localStorage, indexedDB: storage.indexedDB });
originsToSave.delete(origin);
} catch {
}
}
if (originsToSave.size) {
const page = await this.newPage(
progress,
true
/* forStorageState */
);
try {
await page.addRequestInterceptor(progress, (route) => {
route.fulfill({ body: "<html></html>" }).catch(() => {
});
}, "prepend");
for (const origin of originsToSave) {
const frame = page.mainFrame();
await frame.gotoImpl(progress, origin, {});
const storage = await progress.race(frame.evaluateExpression(collectScript, { world: "utility" }));
if (storage.localStorage.length || storage.indexedDB?.length)
result.origins.push({ origin, localStorage: storage.localStorage, indexedDB: storage.indexedDB });
}
} finally {
await page.close();
}
}
return result;
}
isCreatingStorageStatePage() {
return this._creatingStorageStatePage;
}
async setStorageState(progress, state, mode) {
let page;
let interceptor;
try {
if (mode !== "initial") {
await progress.race(this.clearCache());
await progress.race(this.doClearCookies());
}
if (state?.cookies)
await progress.race(this.addCookies(state.cookies));
const newOrigins = new Map(state?.origins?.map((p) => [p.origin, p]) || []);
const allOrigins = /* @__PURE__ */ new Set([...this._origins, ...newOrigins.keys()]);
if (allOrigins.size) {
if (mode === "resetForReuse")
page = this.pages()[0];
if (!page)
page = await this.newPage(
progress,
mode !== "resetForReuse"
/* forStorageState */
);
interceptor = (route) => {
route.fulfill({ body: "<html></html>" }).catch(() => {
});
};
await page.addRequestInterceptor(progress, interceptor, "prepend");
for (const origin of allOrigins) {
const frame = page.mainFrame();
await frame.gotoImpl(progress, origin, {});
const restoreScript = `(() => {
const module = {};
${rawStorageSource.source}
const script = new (module.exports.StorageScript())(${this._browser.options.name === "firefox"});
return script.restore(${JSON.stringify(newOrigins.get(origin))});
})()`;
await progress.race(frame.evaluateExpression(restoreScript, { world: "utility" }));
}
}
this._origins = /* @__PURE__ */ new Set([...newOrigins.keys()]);
} catch (error) {
(0, import_stackTrace.rewriteErrorMessage)(error, `Error setting storage state:
` + error.message);
throw error;
} finally {
if (mode !== "resetForReuse")
await page?.close();
else if (interceptor)
await page?.removeRequestInterceptor(interceptor);
}
}
async extendInjectedScript(source, arg) {
const installInFrame = (frame) => frame.extendInjectedScript(source, arg).catch(() => {
});
const installInPage = (page) => {
page.on(import_page2.Page.Events.InternalFrameNavigatedToNewDocument, installInFrame);
return Promise.all(page.frames().map(installInFrame));
};
this.on(BrowserContext.Events.Page, installInPage);
return Promise.all(this.pages().map(installInPage));
}
async safeNonStallingEvaluateInAllFrames(expression, world, options = {}) {
await Promise.all(this.pages().map((page) => page.safeNonStallingEvaluateInAllFrames(expression, world, options)));
}
harStart(page, options) {
const harId = (0, import_crypto.createGuid)();
this._harRecorders.set(harId, new import_harRecorder.HarRecorder(this, page, options));
return harId;
}
async harExport(harId) {
const recorder = this._harRecorders.get(harId || "");
return recorder.export();
}
addRouteInFlight(route) {
this._routesInFlight.add(route);
}
removeRouteInFlight(route) {
this._routesInFlight.delete(route);
}
async notifyRoutesInFlightAboutRemovedHandler(handler) {
await Promise.all([...this._routesInFlight].map((route) => route.removeHandler(handler)));
}
}
function validateBrowserContextOptions(options, browserOptions) {
if (options.noDefaultViewport && options.deviceScaleFactor !== void 0)
throw new Error(`"deviceScaleFactor" option is not supported with null "viewport"`);
if (options.noDefaultViewport && !!options.isMobile)
throw new Error(`"isMobile" option is not supported with null "viewport"`);
if (options.acceptDownloads === void 0 && browserOptions.name !== "electron")
options.acceptDownloads = "accept";
else if (options.acceptDownloads === void 0 && browserOptions.name === "electron")
options.acceptDownloads = "internal-browser-default";
if (!options.viewport && !options.noDefaultViewport)
options.viewport = { width: 1280, height: 720 };
if (options.proxy)
options.proxy = normalizeProxySettings(options.proxy);
verifyGeolocation(options.geolocation);
}
function verifyGeolocation(geolocation) {
if (!geolocation)
return;
geolocation.accuracy = geolocation.accuracy || 0;
const { longitude, latitude, accuracy } = geolocation;
if (longitude < -180 || longitude > 180)
throw new Error(`geolocation.longitude: precondition -180 <= LONGITUDE <= 180 failed.`);
if (latitude < -90 || latitude > 90)
throw new Error(`geolocation.latitude: precondition -90 <= LATITUDE <= 90 failed.`);
if (accuracy < 0)
throw new Error(`geolocation.accuracy: precondition 0 <= ACCURACY failed.`);
}
function verifyClientCertificates(clientCertificates) {
if (!clientCertificates)
return;
for (const cert of clientCertificates) {
if (!cert.origin)
throw new Error(`clientCertificates.origin is required`);
if (!cert.cert && !cert.key && !cert.passphrase && !cert.pfx)
throw new Error("None of cert, key, passphrase or pfx is specified");
if (cert.cert && !cert.key)
throw new Error("cert is specified without key");
if (!cert.cert && cert.key)
throw new Error("key is specified without cert");
if (cert.pfx && (cert.cert || cert.key))
throw new Error("pfx is specified together with cert, key or passphrase");
}
}
function normalizeProxySettings(proxy) {
let { server, bypass } = proxy;
let url;
try {
url = new URL(server);
if (!url.host || !url.protocol)
url = new URL("http://" + server);
} catch (e) {
url = new URL("http://" + server);
}
if (url.protocol === "socks4:" && (proxy.username || proxy.password))
throw new Error(`Socks4 proxy protocol does not support authentication`);
if (url.protocol === "socks5:" && (proxy.username || proxy.password))
throw new Error(`Browser does not support socks5 proxy authentication`);
server = url.protocol + "//" + url.host;
if (bypass)
bypass = bypass.split(",").map((t) => t.trim()).join(",");
return { ...proxy, server, bypass };
}
function calculateUserAgentEmulation(options) {
const ua = options.userAgent;
if (!ua)
return { navigatorPlatform: void 0, userAgentMetadata: void 0 };
const userAgentMetadata = {
mobile: !!options.isMobile,
model: "",
architecture: "x86",
platform: "Windows",
platformVersion: ""
};
const androidMatch = ua.match(/Android (\d+(\.\d+)?(\.\d+)?)/);
const iPhoneMatch = ua.match(/iPhone OS (\d+(_\d+)?)/);
const iPadMatch = ua.match(/iPad; CPU OS (\d+(_\d+)?)/);
const macOSMatch = ua.match(/Mac OS X (\d+(_\d+)?(_\d+)?)/);
const windowsMatch = ua.match(/Windows\D+(\d+(\.\d+)?(\.\d+)?)/);
if (androidMatch) {
userAgentMetadata.platform = "Android";
userAgentMetadata.platformVersion = androidMatch[1];
userAgentMetadata.architecture = "arm";
} else if (iPhoneMatch) {
userAgentMetadata.platform = "iOS";
userAgentMetadata.platformVersion = iPhoneMatch[1].replace(/_/g, ".");
userAgentMetadata.architecture = "arm";
} else if (iPadMatch) {
userAgentMetadata.platform = "iOS";
userAgentMetadata.platformVersion = iPadMatch[1].replace(/_/g, ".");
userAgentMetadata.architecture = "arm";
} else if (macOSMatch) {
userAgentMetadata.platform = "macOS";
userAgentMetadata.platformVersion = macOSMatch[1].replace(/_/g, ".");
if (!ua.includes("Intel"))
userAgentMetadata.architecture = "arm";
} else if (windowsMatch) {
userAgentMetadata.platform = "Windows";
userAgentMetadata.platformVersion = windowsMatch[1];
} else if (ua.toLowerCase().includes("linux")) {
userAgentMetadata.platform = "Linux";
}
if (ua.includes("ARM") || ua.includes("aarch64"))
userAgentMetadata.architecture = "arm";
let navigatorPlatform;
if (!process.env.PLAYWRIGHT_NO_UA_PLATFORM) {
switch (userAgentMetadata.platform) {
case "Android":
navigatorPlatform = userAgentMetadata.architecture === "arm" ? "Linux armv8l" : "Linux x86_64";
break;
case "iOS":
navigatorPlatform = ua.includes("iPad") ? "iPad" : "iPhone";
break;
case "macOS":
navigatorPlatform = "MacIntel";
break;
case "Linux":
navigatorPlatform = userAgentMetadata.architecture === "arm" ? "Linux aarch64" : "Linux x86_64";
break;
case "Windows":
navigatorPlatform = "Win32";
break;
}
}
return { navigatorPlatform, userAgentMetadata };
}
const paramsThatAllowContextReuse = [
"colorScheme",
"forcedColors",
"reducedMotion",
"contrast",
"screen",
"userAgent",
"viewport",
"testIdAttributeName"
];
const defaultNewContextParamValues = {
noDefaultViewport: false,
ignoreHTTPSErrors: false,
javaScriptEnabled: true,
bypassCSP: false,
offline: false,
isMobile: false,
hasTouch: false,
acceptDownloads: "accept",
strictSelectors: false,
serviceWorkers: "allow",
locale: "en-US"
};
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
BrowserContext,
calculateUserAgentEmulation,
normalizeProxySettings,
validateBrowserContextOptions,
verifyClientCertificates,
verifyGeolocation
});