@agimon-ai/browse-tool 0.10.0 → 0.10.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.cjs +1 -1
- package/dist/cli.mjs +1 -1
- package/dist/index.cjs +1 -1
- package/dist/index.mjs +1 -1
- package/dist/{playwright-test-CZJRdMkJ.cjs → playwright-test-D--wiuvm.cjs} +1 -1
- package/dist/streamable-http-B4RiCNHp.mjs +14 -0
- package/dist/streamable-http-CHbht6Hk.cjs +14 -0
- package/dist/stubs/playwright-test.cjs +1 -1
- package/package.json +5 -5
- package/dist/streamable-http-C82CVPKg.cjs +0 -14
- package/dist/streamable-http-CuNMjfum.mjs +0 -14
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import{a as e,c as t,h as n,p as r,s as i,t as a,v as o,y as s}from"./playwright-test-DlT1Ad7l.mjs";import{stripTypeScriptTypes as c}from"node:module";import"reflect-metadata/lite";import{DEFAULT_PORT_RANGE as l,DEFAULT_REGISTRY_PATH as u,PortRegistryService as d,PortRegistryStateSchema as f,normalizeRepositoryPath as p}from"@agimon-ai/foundation-port-registry";import{ProcessRegistryService as m}from"@agimon-ai/foundation-process-registry";import{Container as h,ContainerModule as g,inject as _,injectable as v,optional as y}from"inversify";import*as b from"node:path";import x,{basename as S,dirname as C,isAbsolute as w,join as T,relative as E,resolve as D}from"node:path";import O,{constants as ee,createWriteStream as te,existsSync as k,mkdirSync as ne,readFileSync as re,writeFileSync as ie}from"node:fs";import*as ae from"node:fs/promises";import{access as oe,chmod as se,mkdir as A,readFile as j,readdir as ce,rename as le,rm as ue,stat as de,symlink as fe,unlink as pe,writeFile as M}from"node:fs/promises";import N,{arch as me,homedir as he,platform as ge,tmpdir as P}from"node:os";import{execFile as _e,execSync as ve,spawn as F}from"node:child_process";import{fileURLToPath as ye}from"node:url";import be,{chromium as xe,firefox as Se,webkit as Ce}from"playwright";import{Readable as we}from"node:stream";import{pipeline as Te}from"node:stream/promises";import{z as I}from"zod";import{AsyncLocalStorage as Ee}from"node:async_hooks";import{createNodeTelemetry as De}from"@agimon-ai/log-sink-mcp";import{SpanKind as Oe,SpanStatusCode as ke,context as Ae,trace as je}from"@opentelemetry/api";import{OTLPTraceExporter as Me}from"@opentelemetry/exporter-trace-otlp-http";import{resourceFromAttributes as Ne}from"@opentelemetry/resources";import{BatchSpanProcessor as Pe,NodeTracerProvider as Fe,SimpleSpanProcessor as Ie}from"@opentelemetry/sdk-trace-node";import{promisify as Le}from"node:util";import{parse as Re}from"yaml";import{build as ze}from"esbuild";import{randomUUID as L}from"node:crypto";import{isInitializeRequest as Be}from"@modelcontextprotocol/sdk/types.js";import{StdioServerTransport as Ve}from"@modelcontextprotocol/sdk/server/stdio.js";import{createServer as He}from"node:http";import{StreamableHTTPServerTransport as Ue}from"@modelcontextprotocol/sdk/server/streamableHttp.js";const R={ProfileService:Symbol.for(`ProfileService`),ProxyConfigService:Symbol.for(`ProxyConfigService`),PageRegistry:Symbol.for(`PageRegistry`),BrowserService:Symbol.for(`BrowserService`),ElementLocatorService:Symbol.for(`ElementLocatorService`),PageMonitorService:Symbol.for(`PageMonitorService`),PauseController:Symbol.for(`PauseController`),AutomationRunner:Symbol.for(`AutomationRunner`),SpecRunner:Symbol.for(`SpecRunner`),SpecBundlerService:Symbol.for(`SpecBundlerService`),SpecDiscoveryService:Symbol.for(`SpecDiscoveryService`),SpecMetadataService:Symbol.for(`SpecMetadataService`),SetupRunner:Symbol.for(`SetupRunner`),WebServerManager:Symbol.for(`WebServerManager`),Logger:Symbol.for(`Logger`),TelemetryService:Symbol.for(`TelemetryService`),CodeSnippetService:Symbol.for(`CodeSnippetService`),HttpServerHealthCheck:Symbol.for(`HttpServerHealthCheck`),HttpServerManager:Symbol.for(`HttpServerManager`),HttpBrowserClient:Symbol.for(`HttpBrowserClient`),RemoteToolExecutor:Symbol.for(`RemoteToolExecutor`),ExtensionTaskQueue:Symbol.for(`ExtensionTaskQueue`),ExtensionToolDelegator:Symbol.for(`ExtensionToolDelegator`),ToolExecutor:Symbol.for(`ToolExecutor`),StealthLauncher:Symbol.for(`StealthLauncher`),BrowserLockManager:Symbol.for(`BrowserLockManager`),ExtensionSessionRegistry:Symbol.for(`ExtensionSessionRegistry`),ExtensionPageProxy:Symbol.for(`ExtensionPageProxy`),ExtensionSpecRunner:Symbol.for(`ExtensionSpecRunner`),McpSessionTracker:Symbol.for(`McpSessionTracker`),ChromeForTestingService:Symbol.for(`ChromeForTestingService`),WebSocketHub:Symbol.for(`WebSocketHub`),IdleCleanupService:Symbol.for(`IdleCleanupService`),PortRegistryService:Symbol.for(`PortRegistryService`),McpPortAllocationService:Symbol.for(`McpPortAllocationService`),ProcessRegistryService:Symbol.for(`ProcessRegistryService`),Tool:Symbol.for(`Tool`)};function z(e,t){if(typeof Reflect==`object`&&typeof Reflect.metadata==`function`)return Reflect.metadata(e,t)}function B(e,t){return function(n,r){t(n,r,e)}}function V(e,t,n,r){var i=arguments.length,a=i<3?t:r===null?r=Object.getOwnPropertyDescriptor(t,n):r,o;if(typeof Reflect==`object`&&typeof Reflect.decorate==`function`)a=Reflect.decorate(e,t,n,r);else for(var s=e.length-1;s>=0;s--)(o=e[s])&&(a=(i<3?o(a):i>3?o(t,n,a):o(t,n))||a);return i>3&&a&&Object.defineProperty(t,n,a),a}let We=class{sessions=new Map;sessionIdCounter=0;constructor(e,t,n,r,i,a){this.browserService=e,this.specRunner=t,this.specMetadataService=n,this.setupRunner=r,this.webServerManager=i,this.pageRegistry=a}async runSpec(e){let{specPath:t,automationId:n,keepBrowserOpen:r=!1,browserOptions:i={},browserId:a,pageId:o,hooksPath:s}=e,c=n??`spec-${++this.sessionIdCounter}`,l={id:c,scriptPath:t,status:`pending`,pageIds:[],browserId:null,error:null,createdAt:new Date,updatedAt:new Date};this.sessions.set(c,l);let u=!!(a&&o);try{u&&(l.browserId=a,l.pageIds.push(o)),l.status=`running`,l.updatedAt=new Date;let e=await this.specRunner.executeSpec({specPath:t,sessionId:c,browserType:i.browserType,headless:i.headless,hooksPath:s,baseURL:i.baseURL,keepBrowserOpen:r});if(r&&e.browser){let t=this.browserService.registerExternalBrowser(e.browser);if(l.browserId=t,e.keptPages?.length)for(let n of e.keptPages){let e=await this.pageRegistry.register({page:n.page,context:n.context,browserId:t,onClose:e=>{let t=l.pageIds.indexOf(e);t>=0&&l.pageIds.splice(t,1)}});l.pageIds.push(e)}}return l.status=`completed`,l.updatedAt=new Date,{automationId:c,status:`completed`,pageIds:r||u?[...l.pageIds]:[],browserId:r||u?l.browserId:null,specResult:e}}catch(e){return l.status=`failed`,l.error=e instanceof Error?e.message:String(e),l.updatedAt=new Date,{automationId:c,status:`failed`,pageIds:u?[...l.pageIds]:[],browserId:u?l.browserId:null}}}async runSpecEnhanced(e){let{specPath:t,automationId:n,keepBrowserOpen:r=!1,browserOptions:i={},testFilter:a,specArgs:o,configPath:s,runSetup:c=!1,startServer:l=!1,stopServerAfter:u=!0,browserId:d,pageId:f,hooksPath:p}=e,m=n??`spec-${++this.sessionIdCounter}`,h={id:m,scriptPath:t,status:`pending`,pageIds:[],browserId:null,error:null,createdAt:new Date,updatedAt:new Date};this.sessions.set(m,h);let g,_;try{let e=s?await this.specMetadataService.loadPlaywrightConfig(s):null,n=s?C(s):process.cwd();l&&e?.webServer&&(g=await this.webServerManager.startServer(e.webServer,n));let v=!!(d&&f);if(v&&(h.browserId=d,h.pageIds.push(f)),h.status=`running`,h.updatedAt=new Date,c&&e?.setupFile){let t=e.setupFile.startsWith(`/`)?e.setupFile:`${n}/${e.setupFile}`,r=e.setupExport??`default`,a=await this.browserService.launch(i);try{if(_=await this.setupRunner.runSetup(t,r,{browser:a.browserInstance.browser,context:a.browserInstance.context,page:a.page}),!_.success)throw Error(_.error??`Setup failed`)}finally{await this.browserService.closeBrowser(a.browserInstance.id)}}let y=await this.specRunner.executeSpecEnhanced({specPath:t,sessionId:m,browserType:i.browserType,headless:i.headless,testFilter:a,specArgs:{...o,..._?.state},hooksPath:p,baseURL:i.baseURL,keepBrowserOpen:r});if(r&&y.browser){let e=this.browserService.registerExternalBrowser(y.browser);if(h.browserId=e,y.keptPages?.length)for(let t of y.keptPages){let n=await this.pageRegistry.register({page:t.page,context:t.context,browserId:e,onClose:e=>{let t=h.pageIds.indexOf(e);t>=0&&h.pageIds.splice(t,1)}});h.pageIds.push(n)}}return h.status=`completed`,h.updatedAt=new Date,u&&g?.started&&await this.webServerManager.stopServer(),{automationId:m,status:`completed`,pageIds:r||v?[...h.pageIds]:[],browserId:r||v?h.browserId:null,specResult:y,serverResult:g,setupResult:_}}catch(e){h.status=`failed`,h.error=e instanceof Error?e.message:String(e),h.updatedAt=new Date,u&&g?.started&&await this.webServerManager.stopServer();let t=!!(d&&f);return{automationId:m,status:`failed`,pageIds:t?[...h.pageIds]:[],browserId:t?h.browserId:null,serverResult:g,setupResult:_}}}getSession(e){return this.sessions.get(e)}listSessions(){return Array.from(this.sessions.values())}async stop(e){let t=this.sessions.get(e);return t?(t.browserId&&await this.browserService.closeBrowser(t.browserId),t.status=`completed`,t.updatedAt=new Date,!0):!1}};We=V([v(),B(0,_(R.BrowserService)),B(1,_(R.SpecRunner)),B(2,_(R.SpecMetadataService)),B(3,_(R.SetupRunner)),B(4,_(R.WebServerManager)),B(5,_(R.PageRegistry)),z(`design:paramtypes`,[Object,Object,Object,Object,Object,Object])],We);var Ge=class extends Error{constructor(e,t,n){super(e),this.pid=t,this.userDataDir=n,this.name=`BrowserSessionConflictError`}};let Ke=class{platform=N.platform();async checkLock(e){let t={isLocked:!1};return this.platform===`win32`?this.checkWindowsLock(e,t):this.checkUnixLock(e,t)}async cleanupOrphanedLocks(e){let t=await this.checkLock(e);if(!t.isLocked)return!0;if(t.isProcessAlive)return!1;let n=[x.join(e,`SingletonLock`),x.join(e,`SingletonCookie`),x.join(e,`SingletonSocket`),x.join(e,`lockfile`)];for(let e of n)try{O.existsSync(e)&&await pe(e)}catch{}let r=x.join(e,`Default`,`LOCK`);try{O.existsSync(r)&&await pe(r)}catch{}return!0}async ensureAvailable(e){if(!O.existsSync(e))return;let t=await this.checkLock(e);if(t.isLocked){if(t.isProcessAlive)throw new Ge(`Browser session conflict: Another Chrome process (PID: ${t.pid}) is using this profile.\nUser data directory: ${e}\n\nTo resolve:\n1. Close the existing browser session manually\n2. Or use a different profile name\n3. Or kill the process: kill ${t.pid}`,t.pid,e);if(!await this.cleanupOrphanedLocks(e))throw Error(`Failed to clean up orphaned browser lock in ${e}`)}}async checkUnixLock(e,t){let n=x.join(e,`SingletonLock`),r=x.join(e,`SingletonSocket`);if(t.lockFilePath=n,t.socketPath=r,this.symlinkExists(n))try{let e=O.readlinkSync(n).match(/-(\d+)$/);e&&(t.isLocked=!0,t.pid=Number.parseInt(e[1],10),t.isProcessAlive=this.isProcessRunning(t.pid))}catch{t.isLocked=!0,t.isProcessAlive=O.existsSync(r)}return!t.isLocked&&O.existsSync(r)&&(t.isLocked=!0,t.isProcessAlive=!0),t}symlinkExists(e){try{return O.lstatSync(e).isSymbolicLink()}catch{return!1}}async checkWindowsLock(e,t){let n=x.join(e,`lockfile`);if(t.lockFilePath=n,O.existsSync(n))try{let e=await j(n,`utf-8`),r=Number.parseInt(e.trim(),10);Number.isNaN(r)||(t.isLocked=!0,t.pid=r,t.isProcessAlive=this.isProcessRunning(r))}catch{t.isLocked=!0,t.isProcessAlive=!0}return t}isProcessRunning(e){try{return process.kill(e,0),!0}catch(e){return e.code===`EPERM`}}};Ke=V([v()],Ke);var qe=`147.0.7727.50`,Je=`browse-tool/chrome-for-testing:147.0.7727.50`,Ye=`/usr/local/bin/google-chrome-for-testing`,Xe=`linux/amd64`,Ze=`browse-tool/cloakbrowser:latest`,Qe=`/usr/local/bin/cloakbrowser`,$e=`linux/amd64`;const H=18e4;var et;const tt=qe,nt=`BROWSE_TOOL_CHROME_FOR_TESTING_PATH`;function rt(e,t){let n=e.split(`.`).map(e=>Number.parseInt(e,10)||0),r=t.split(`.`).map(e=>Number.parseInt(e,10)||0),i=Math.max(n.length,r.length);for(let e=0;e<i;e+=1){let t=n[e]??0,i=r[e]??0;if(t!==i)return t>i?1:-1}return 0}function it(){let e=process.env.BROWSE_TOOL_CHROME_FOR_TESTING_VERSION?.trim();return e&&e.length>0?e:tt}function at(){let e=process.env[nt]?.trim();return e&&e.length>0?e:void 0}function ot(e){return rt(e,`147.0.7727.3`)>=0}function st(){return ot(it())}let ct=class{static{et=this}static API_URL=`https://googlechromelabs.github.io/chrome-for-testing/known-good-versions-with-downloads.json`;cacheDir;constructor(){this.cacheDir=T(he(),`.cache`,`chrome-for-testing`)}async getExecutablePath(){let e=at();if(e)return this.validateConfiguredExecutablePath(e);let t=this.getExpectedExecutablePath(),n=await this.getInstalledVersion();try{let e=this.getRequestedVersion(),r=await this.fetchVersionInfo(e);return k(t)&&n===e?t:this.download(r)}catch(e){if(k(t))return t;throw e}}async isInstalled(){return k(this.getExpectedExecutablePath())}async getInstalledVersion(){let e=T(this.cacheDir,`version.txt`);if(!k(e))return null;try{return(await j(e,`utf-8`)).trim()}catch{return null}}async update(){let e=T(this.cacheDir,`chrome`);return k(e)&&await ue(e,{recursive:!0,force:!0}),this.download(await this.fetchVersionInfo(this.getRequestedVersion()))}async download(e){ne(this.cacheDir,{recursive:!0});let t=this.getDownloadUrl(e);if(!t)throw Error(`No Chrome for Testing download available for ${this.getPlatformKey()}`);let n=T(this.cacheDir,`chrome.zip`),r=T(this.cacheDir,`chrome`);await this.downloadFile(t,n),await this.extractZip(n,r),await ue(n,{force:!0}),await M(T(this.cacheDir,`version.txt`),e.version,`utf-8`);let i=this.getExpectedExecutablePath();if(!k(i))throw Error(`Chrome for Testing executable not found at expected path: ${i}`);return ge()!==`win32`&&await se(i,493),i}async fetchVersionInfo(e){let t=await fetch(et.API_URL);if(!t.ok)throw Error(`Failed to fetch Chrome for Testing info: ${t.statusText}`);let n=(await t.json()).versions.find(t=>t.version===e);if(!n)throw Error(`Chrome for Testing version ${e} not found in feed`);return n}getRequestedVersion(){return it()}async validateConfiguredExecutablePath(e){try{if(!(await de(e)).isFile())throw Error(`configured path is not a file`);return ge()!==`win32`&&await oe(e,ee.X_OK),e}catch(t){throw Error(`Chrome for Testing executable configured by ${nt} was not found or is not executable: ${e}`,{cause:t})}}getDownloadUrl(e){let t=this.getPlatformKey();return e.downloads.chrome.find(e=>e.platform===t)?.url}getPlatformKey(){let e=ge(),t=me();if(e===`darwin`)return t===`arm64`?`mac-arm64`:`mac-x64`;if(e===`linux`)return`linux64`;if(e===`win32`)return t===`x64`?`win64`:`win32`;throw Error(`Unsupported platform: ${e}`)}getExpectedExecutablePath(){let e=ge(),t=T(this.cacheDir,`chrome`);if(e===`darwin`)return T(t,`chrome-mac-${me()===`arm64`?`arm64`:`x64`}`,`Google Chrome for Testing.app`,`Contents`,`MacOS`,`Google Chrome for Testing`);if(e===`linux`)return T(t,`chrome-linux64`,`chrome`);if(e===`win32`)return T(t,`chrome-win${me()===`x64`?`64`:`32`}`,`chrome.exe`);throw Error(`Unsupported platform: ${e}`)}async downloadFile(e,t){let n=await fetch(e);if(!n.ok||!n.body)throw Error(`Failed to download: ${n.statusText}`);let r=`${t}.tmp`,i=te(r),a=n.body.getReader();await Te(new we({async read(){let{done:e,value:t}=await a.read();e?this.push(null):this.push(Buffer.from(t))}}),i),await le(r,t)}async extractZip(e,t){let n=ge();return new Promise((r,i)=>{let a;n===`win32`?a=F(`powershell`,[`-NoProfile`,`-Command`,`Expand-Archive -Path "${e}" -DestinationPath "${t}" -Force`]):(ne(t,{recursive:!0}),a=F(`unzip`,[`-o`,`-q`,e,`-d`,t])),a.on(`close`,e=>{e===0?r():i(Error(`Failed to extract zip (exit code: ${e})`))}),a.on(`error`,e=>{i(Error(`Failed to spawn extraction process: ${e.message}`))})})}};ct=et=V([v(),z(`design:paramtypes`,[])],ct);const lt=[`pnpm-workspace.yaml`,`nx.json`,`.git`],ut=Je,dt=Xe,ft=ut.split(`:`)[0]??ut,pt=Ze,mt=pt.split(`:`)[0]??pt;function ht(e=process.cwd()){let t=x.resolve(e);for(;;){for(let e of lt)if(gt(x.join(t,e)))return t;let n=x.dirname(t);if(n===t)return e;t=n}}function gt(e){return O.existsSync(e)}function _t(){return ft}function vt(e){return`${_t()}:${e}`}function yt(e){let t=x.resolve(e);for(let e=0;e<8;e+=1){let e=x.join(t,`package.json`);if(gt(e))try{if(JSON.parse(O.readFileSync(e,`utf8`)).name===`@agimon-ai/browse-tool`)return t}catch{}let n=x.dirname(t);if(n===t)break;t=n}return x.resolve(e,`../..`)}function bt(){let e=x.dirname(ye(import.meta.url));return x.join(yt(e),`docker/chrome-for-testing.Dockerfile`)}function xt(){return ht(yt(x.dirname(ye(import.meta.url))))}function St(e){if(!e?.includes(`:`))return;let[,t]=e.split(`:`,2),n=t?.trim();if(!(!n||!/^\d+\.\d+\.\d+\.\d+$/.test(n)))return n}function Ct(e){return(e?.trim()||dt).toLowerCase()}function wt(e){let t=Ct(e);return t.includes(`arm64`)||t.includes(`aarch64`)?`linux-arm64`:`linux64`}function Tt(e){return e.startsWith(`${_t()}:`)}async function Et(e,t){return await new Promise((n,r)=>{let i=``,a=F(`docker`,e,{stdio:t===`pipe`?[`ignore`,`pipe`,`pipe`]:[`ignore`,t,t]});t===`pipe`&&a.stderr?.on(`data`,e=>{i+=e.toString(`utf8`)}),a.once(`error`,r),a.once(`close`,e=>{n({exitCode:e??1,stderr:i})})})}async function Dt(e){return(await Et([`image`,`inspect`,e],`ignore`)).exitCode===0}async function Ot(e={}){let t=e.version?.trim()||St(e.image)||it(),n=e.image?.trim()||vt(t),r=Ct(e.platform),i=wt(r),a=e.stdio??`inherit`,o=await Et([`buildx`,`build`,`--platform`,r,`--load`,`-t`,n,`--build-arg`,`CFT_VERSION=${t}`,`--build-arg`,`CFT_ARCHIVE_PLATFORM=${i}`,`-f`,bt(),xt()],a);if(o.exitCode!==0){let e=o.stderr.trim();throw Error(e?`Failed to build Docker image ${n}: ${e}`:`Failed to build Docker image ${n}`)}return{image:n,version:t,platform:r}}async function kt(e){return!Tt(e.image)||await Dt(e.image)?!1:(await Ot({image:e.image,platform:e.platform,stdio:`inherit`}),!0)}function At(e){return e.startsWith(`${mt}:`)}function jt(){let e=x.dirname(ye(import.meta.url));return x.join(yt(e),`docker/cloakbrowser.Dockerfile`)}async function Mt(e={}){let t=e.image?.trim()||pt,n=Ct(e.platform),r=e.stdio??`inherit`,i=await Et([`buildx`,`build`,`--platform`,n,`--load`,`-t`,t,`-f`,jt(),xt()],r);if(i.exitCode!==0){let e=i.stderr.trim();throw Error(e?`Failed to build CloakBrowser Docker image ${t}: ${e}`:`Failed to build CloakBrowser Docker image ${t}`)}return{image:t,platform:n}}async function Nt(e){return!At(e.image)||await Dt(e.image)?!1:(await Mt({image:e.image,platform:e.platform,stdio:`inherit`}),!0)}function Pt(){let e=new Map,t=ve(`ps -axo pid=,ppid=`,{encoding:`utf-8`,stdio:[`ignore`,`pipe`,`ignore`],timeout:3e3});for(let n of t.split(`
|
|
2
|
+
`)){let[t,r]=n.trim().split(/\s+/,2);if(!t||!r)continue;let i=Number(t),a=Number(r);if(Number.isNaN(i)||Number.isNaN(a))continue;let o=e.get(a);o?o.push(i):e.set(a,[i])}return e}function Ft(e){let t=[];try{let n;try{n=ve(`pgrep -P ${e}`,{encoding:`utf-8`,stdio:[`ignore`,`pipe`,`ignore`],timeout:3e3}).trim().split(`
|
|
3
|
+
`).filter(Boolean).map(Number).filter(e=>!Number.isNaN(e))}catch{n=Pt().get(e)??[]}for(let e of n)t.push(...Ft(e)),t.push(e)}catch{}return t}function It(e){try{return process.kill(e,0),!0}catch{return!1}}async function Lt(e,t=3e3){if(!It(e))return;let n=[...Ft(e),e];for(let e of n)try{process.kill(e,`SIGTERM`)}catch{}let r=Date.now()+t;for(;Date.now()<r;){if(!n.some(It))return;await new Promise(e=>setTimeout(e,100))}for(let e of n)try{It(e)&&process.kill(e,`SIGKILL`)}catch{}}function Rt(e,t){let n=x.join(N.tmpdir(),`proxy-auth-ext-${Date.now()}-${Math.random().toString(36).slice(2,8)}`);O.mkdirSync(n,{recursive:!0});let r={manifest_version:3,name:`Proxy Auth Helper`,version:`1.0.0`,permissions:[`webRequest`,`webRequestAuthProvider`],host_permissions:[`<all_urls>`],background:{service_worker:`background.js`}},i=[`chrome.webRequest.onAuthRequired.addListener(`,` (details) => ({ authCredentials: { username: ${JSON.stringify(e)}, password: ${JSON.stringify(t)} } }),`,` { urls: ["<all_urls>"] },`,` ["blocking"]`,`);`,``].join(`
|
|
4
|
+
`);return O.writeFileSync(x.join(n,`manifest.json`),JSON.stringify(r,null,2)),O.writeFileSync(x.join(n,`background.js`),i),n}function zt(e){let t=[`--proxy-server=${e.server}`];return e.bypass&&t.push(`--proxy-bypass-list=${e.bypass}`),t}var U;const Bt=new Set([`docker`,`docker-vm`,`docker-chromium`,`docker-chrome-testing`,`docker-cloakbrowser`]),Vt=Je,Ht=Ye,Ut=Ze,Wt=Qe,Gt=4e3,Kt=process.env.BROWSE_TOOL_DEBUG_EXTENSION_RECORDING===`1`,qt=[`Current Session`,`Current Tabs`,`Last Session`,`Last Tabs`,`Sessions`],Jt=new Set([`Default`,`Guest Profile`,`System Profile`]);let Yt=class{static{U=this}static MAX_ACTIVE_BROWSERS=15;static MAX_PAGES_PER_BROWSER=15;static chromeForTestingLaunchQueue=Promise.resolve();browsers=new Map;closingExtensionRecordings=new Map;profileToBrowserId=new Map;browserIdCounter=0;constructor(e,t,n,r,i,a,o,s){this.profileService=e,this.pageRegistry=t,this.lockManager=n,this.chromeForTesting=r,this.pageMonitor=i,this.webSocketHub=a,this.extensionSessionRegistry=o,this.extensionTaskQueue=s}getBrowserByProfile(e){let t=this.profileToBrowserId.get(e);if(t){let n=this.browsers.get(t);if(n){if(this.isBrowserAlive(n))return n;this.cleanupStaleBrowser(t,e)}else this.profileToBrowserId.delete(e)}}async launch(e={}){let{profileName:t,browserType:n=`chromium`,headless:r=!0,onPageClose:i,baseURL:a,recordVideo:o,proxy:s,browserId:c}=e;if(t){let e=this.getBrowserByProfile(t);if(e){let{pageId:t,page:n}=await this.newPage(e.id,i);return{browserInstance:e,pageId:t,page:n}}}await this.enforceMaxBrowserLimit();let l=c||(t?`browser-${t}`:`browser-${++this.browserIdCounter}`);if(c&&this.browsers.has(c))throw Error(`Browser ID "${c}" is already in use`);let u=this.getBrowserType(n),d=t?await this.profileService.get(t):null;t&&!d&&(d=await this.profileService.create({name:t,browserType:n}));let f=s?{proxy:{server:s.server,username:s.username,password:s.password,bypass:s.bypass}}:{},p=process.platform===`linux`&&typeof process.getuid==`function`&&process.getuid()===0,m=n===`chromium`?process.env.CHROME_BIN||process.env.PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH:void 0,h=process.env.BROWSE_TOOL_DISABLE_GPU===`true`,g=n===`chromium`?[...h?[`--disable-gpu`,`--disable-software-rasterizer`,`--disable-dev-shm-usage`,`--disable-features=VizDisplayCompositor`]:[],...p?[`--no-sandbox`,`--disable-setuid-sandbox`]:[]]:[],_=n===`chromium`?{...g.length>0?{args:g}:{},...m?{executablePath:m}:{}}:{},v;if(!r&&n===`chromium`)try{v=await u.launch({headless:r,...m?{}:{channel:`chrome`},..._,...f})}catch{v=await u.launch({headless:r,..._,...f})}else v=await u.launch({headless:r,..._,...f});let y=this.buildContextOptions(d,a,o);if(d&&t){let e=await this.profileService.loadStorageState(t);e&&(y.storageState=e)}let b=await v.newContext(y),x=await b.newPage(),S=new Date,C={id:l,mode:`playwright`,browser:v,context:b,profileName:t,pageIds:new Set,currentPageId:null,createdAt:S,lastAccessedAt:S};this.browsers.set(l,C),v.on(`disconnected`,()=>{this.cleanupStaleBrowser(l,t)}),t&&this.profileToBrowserId.set(t,l);let w=await this.pageRegistry.register({page:x,browser:v,context:b,browserId:l,profileName:t,onClose:e=>{if(C.pageIds.delete(e),C.currentPageId===e){let e=Array.from(C.pageIds);C.currentPageId=e.length>0?e[0]:null}i?.(e),C.pageIds.size===0&&this.closeBrowser(l).catch(()=>{})}});return this.pageMonitor?.startMonitoring(w,x),C.pageIds.add(w),C.currentPageId=w,{browserInstance:C,pageId:w,page:x}}async launchWithExtension(e={}){let{profileName:t,command:n,commandArgs:r=[],browserId:i,mode:a=`extension`}=e;if(t){let e=this.getBrowserByProfile(t);if(e)if(!this.canReuseExtensionBrowser(e))await this.closeBrowser(e.id).catch(()=>{this.cleanupStaleBrowser(e.id,t)});else{let t=this.getTrackedExtensionPageId(e);return this.recordBrowserActivity(e.id,t),{browserInstance:e,pageId:t,process:{pid:e.pid}}}}await this.enforceMaxBrowserLimit();let o=this.resolveExtensionPath(e.extensionPath),s=x.join(o,`manifest.json`);if(!O.existsSync(s))throw Error(`Extension manifest not found at ${s}. Please build the extension first: pnpm build:extension`);let c=i||(t?`browser-${t}`:`browser-${++this.browserIdCounter}`);if(i&&this.browsers.has(i))throw Error(`Browser ID "${i}" is already in use`);t&&(await this.profileService.get(t)||await this.profileService.create({name:t,browserType:`chromium`}));let l=this.createUserDataDir(t);l&&await this.lockManager.ensureAvailable(l);let u=await this.resolveVmRuntime({command:n,launchMode:a,vmDockerBrowser:e.vmDockerBrowser,extensionHostPath:o,requestedRuntimeExtensionPath:e.extensionPath,userDataDir:l});if(u.kind===`docker`)try{this.ensureDockerWritableUserDataDir(l)}catch(e){let t=e instanceof Error?e.message:String(e);throw Error(`Failed to prepare Docker user-data directory "${l}": ${t}`,{cause:e})}let d=[u.extensionPath],f=[];if(u.kind===`docker`&&f.push(...this.createDockerVolumeArgs(o,u.extensionPath,!0)),e.proxy?.username&&e.proxy?.password){let t=Rt(e.proxy.username,e.proxy.password);if(u.kind===`docker`){let e=`${u.docker.extensionPathInContainer}-proxy-auth-${Date.now()}`;d.push(e),f.push(...this.createDockerVolumeArgs(t,e,!0))}else d.push(t)}let p=d.join(`,`),m=process.env.BROWSE_TOOL_AUTO_SELECT_TAB_CAPTURE_SOURCE_TITLE?.trim(),h=process.env.BROWSE_TOOL_AUTO_SELECT_DESKTOP_CAPTURE_SOURCE?.trim(),g=process.env.BROWSE_TOOL_USE_FAKE_UI_FOR_MEDIA_STREAM===`1`,_=u.kind===`docker`||process.platform===`linux`&&typeof process.getuid==`function`&&process.getuid()===0,v=[`--user-data-dir=${u.userDataDir}`,`--load-extension=${p}`,`--disable-extensions-except=${p}`,`--enable-extensions`,`--auto-accept-this-tab-capture`,...g?[`--use-fake-ui-for-media-stream`]:[],...m?[`--auto-select-tab-capture-source-by-title=${m}`]:[],...h?[`--auto-select-desktop-capture-source=${h}`]:[],`--disable-blink-features=AutomationControlled`,`--disable-infobars`,`--no-first-run`,`--no-default-browser-check`,`--disable-background-timer-throttling`,`--disable-backgrounding-occluded-windows`,`--disable-renderer-backgrounding`,`--disable-component-update`,`--disable-features=TranslateUI`,`--disable-ipc-flooding-protection`,...u.kind===`docker`?[`--disable-gpu`]:[],`--silent-debugger-extension-api`,`--password-store=basic`,`--use-mock-keychain`,`--disable-dev-shm-usage`,`--webrtc-ip-handling-policy=disable_non_proxied_udp`,`--enforce-webrtc-ip-permission-check`,`--lang=${process.env.LANG?.split(`.`)[0]?.replace(`_`,`-`)||`en-US`}`,..._?[`--no-sandbox`,`--disable-setuid-sandbox`]:[],`--window-size=1920,1080`,...e.proxy?zt(e.proxy):[],e.url??`about:blank`],y=e=>{try{let t=new URL(e);return t.protocol===`http:`||t.protocol===`https:`||t.protocol===`file:`}catch{return!1}},b=e=>{if(!y(e))return e;try{let t=new URL(e),n=t.pathname.endsWith(`/`)&&t.pathname!==`/`?t.pathname.slice(0,-1):t.pathname;return`${t.protocol}//${t.host}${n}${t.search}`}catch{return e}},S=b(e.url??`about:blank`),C=a===`vm`&&r.length>0?[...v,...r.filter(e=>y(e)?b(e)!==S:!0)]:r.length>0?r:v,w=u.kind===`docker`?{TZ:Intl.DateTimeFormat().resolvedOptions().timeZone,...e.dockerEnv}:e.dockerEnv,T=u.kind===`docker`?[...this.buildDockerSpawnArgs(u.docker,l,f,e.dockerRunArgs,w,e.dockerEnableVnc),...C]:C,E,D=0,ee;for(let n=1;n<=3;n+=1)try{let n=()=>this.startExtensionBrowser({browserId:c,launchMode:a,profileName:t,vmRuntime:u,spawnArgs:T,recordVideo:e.recordVideo,startupDelayMs:e.startupDelayMs,launchUrl:e.url});return this.shouldSerializeChromeForTestingLaunch(u)?await this.runSerializedChromeForTestingLaunch(n):await n()}catch(e){if(E=e instanceof Error?e:Error(String(e)),!this.isRetryableExtensionLaunchError(E)||n===3){if(u.kind===`docker`&&(D>0||ee)){let e=[D>0?`Docker retry cleanup attempts: ${D}.`:``,ee?`Last Docker retry cleanup: ${ee}.`:``].filter(Boolean).join(` `);E.message=`${E.message} ${e}`.trim()}throw E}u.kind===`docker`&&(D+=1,ee=await this.prepareDockerRetryUserDataDir(l)),await this.delay(250*n)}throw E??Error(`Extension browser launch failed`)}async resolveVmRuntime(e){let{command:t,launchMode:n,vmDockerBrowser:r,extensionHostPath:i,requestedRuntimeExtensionPath:a,userDataDir:o}=e;if(n===`vm`&&this.isDockerVmAlias(t)){let e=this.resolveDockerRuntimeExtensionPath(a),t=process.env.PLAYWRIGHT_VM_DOCKER_USER_DATA_DIR||`/vm/user-data`,n=this.resolveDockerBrowserType(r),{image:i,browserBinary:o,platform:s}=this.resolveDockerImageConfig(n);return n===`chrome-for-testing`?await kt({image:i,platform:s}):await Nt({image:i,platform:s}),{kind:`docker`,command:`docker`,extensionPath:e,userDataDir:t,docker:{image:i,browserBinary:o,platform:s,extensionPathInContainer:e,userDataDirInContainer:t}}}return{kind:`host`,command:await this.resolveLaunchCommand({command:t,launchMode:n,vmDockerBrowser:r}),extensionPath:i,userDataDir:o}}resolveDockerRuntimeExtensionPath(e){return e&&!O.existsSync(x.join(e,`manifest.json`))?e:process.env.PLAYWRIGHT_VM_DOCKER_EXTENSION_PATH||`/vm/extensions/playwright`}resolveDockerBrowserType(e){let t=process.env.PLAYWRIGHT_VM_DOCKER_BROWSER?.trim().toLowerCase();return t===`chrome-for-testing`||t===`cft`?`chrome-for-testing`:t===`cloakbrowser`||t===`cloak`?`cloakbrowser`:e||(process.env.PLAYWRIGHT_VM_DOCKER_IMAGE?`chrome-for-testing`:`cloakbrowser`)}resolveDockerImageConfig(e){return e===`chrome-for-testing`?{image:process.env.PLAYWRIGHT_VM_DOCKER_IMAGE||Vt,browserBinary:process.env.PLAYWRIGHT_VM_DOCKER_BROWSER_BINARY||Ht,platform:process.env.PLAYWRIGHT_VM_DOCKER_PLATFORM||Xe}:{image:process.env.PLAYWRIGHT_VM_DOCKER_IMAGE||Ut,browserBinary:process.env.PLAYWRIGHT_VM_DOCKER_BROWSER_BINARY||Wt,platform:process.env.PLAYWRIGHT_VM_DOCKER_PLATFORM||$e}}createDockerVolumeArgs(e,t,n){return[`-v`,`${x.resolve(e)}:${t}${n?`:ro`:``}`]}ensureDockerWritableUserDataDir(e){O.mkdirSync(e,{recursive:!0,mode:511});let t=[e];for(;t.length>0;){let e=t.pop();if(!e)continue;O.chmodSync(e,511);let n=O.readdirSync(e,{withFileTypes:!0});for(let r of n){let n=x.join(e,r.name);O.chmodSync(n,r.isDirectory()?511:438),r.isDirectory()&&t.push(n)}}}buildDockerSpawnArgs(e,t,n,r,i,a){let o=[`run`,`--rm`,`--init`,`--shm-size`,process.env.PLAYWRIGHT_VM_DOCKER_SHM_SIZE||`2g`];return e.platform&&o.push(`--platform`,e.platform),process.env.PLAYWRIGHT_VM_DOCKER_NETWORK&&o.push(`--network`,process.env.PLAYWRIGHT_VM_DOCKER_NETWORK),process.platform===`linux`&&o.push(`--add-host`,`host.docker.internal:host-gateway`),o.push(...n),o.push(...this.createDockerVolumeArgs(t,e.userDataDirInContainer,!1)),o.push(...this.createDockerEnvArgs(i)),o.push(...this.parseDockerRunArgs(process.env.PLAYWRIGHT_VM_DOCKER_RUN_ARGS)),o.push(...r??[]),a?(o.push(`--entrypoint`,`/bin/bash`,e.image,`-lc`,this.buildDockerBrowserLaunchScript(e.browserBinary,{enableVnc:!0}),`_`),o):(o.push(`--entrypoint`,`/bin/bash`,e.image,`-lc`,this.buildDockerBrowserLaunchScript(e.browserBinary,{enableVnc:!1}),`_`),o)}buildDockerBrowserLaunchScript(e,t){let n=[`set -euo pipefail`,`export XDG_RUNTIME_DIR="/tmp/runtime-browse"`,`mkdir -p "$XDG_RUNTIME_DIR" "$XDG_RUNTIME_DIR/pulse"`,`chmod 700 "$XDG_RUNTIME_DIR" || true`,`command -v dbus-daemon >/dev/null 2>&1 && export DBUS_SESSION_BUS_ADDRESS="$(dbus-daemon --session --fork --print-address --nopidfile)" || true`,`if command -v pulseaudio >/dev/null 2>&1; then pulseaudio --check >/dev/null 2>&1 || pulseaudio --daemonize=yes --exit-idle-time=-1 --disable-shm=true --log-target=stderr >/tmp/pulseaudio.log 2>&1; export PULSE_SERVER="unix:$XDG_RUNTIME_DIR/pulse/native"; fi`,`Xvfb :99 -screen 0 1920x1080x24 -nolisten tcp >/tmp/xvfb.log 2>&1 &`,`if command -v xdpyinfo >/dev/null 2>&1; then for i in $(seq 1 30); do xdpyinfo -display :99 >/dev/null 2>&1 && break; sleep 0.2; done; else sleep 1; fi`,`export DISPLAY=:99`,`BROWSER_BIN=${JSON.stringify(e)}`,`exec "$BROWSER_BIN" "$@"`];return t.enableVnc&&n.splice(n.length-2,0,'if [ -n "${SE_VNC_PASSWORD:-}" ]; then',' x11vnc -storepasswd "${SE_VNC_PASSWORD}" /tmp/vnc-passwd >/dev/null 2>&1',` X11VNC_AUTH="-rfbauth /tmp/vnc-passwd"`,`else`,` X11VNC_AUTH=""`,`fi`,'x11vnc -forever -shared -rfbport "${SE_VNC_PORT:-5900}" -display :99 $X11VNC_AUTH >/tmp/x11vnc.log 2>&1 &',`if command -v websockify >/dev/null 2>&1; then`,' websockify --web /opt/bin/noVNC "${SE_NO_VNC_PORT:-7900}" "localhost:${SE_VNC_PORT:-5900}" >/tmp/novnc.log 2>&1 &',`fi`),n.join(`
|
|
5
|
+
`)}createDockerEnvArgs(e){if(!e)return[];let t=[];for(let[n,r]of Object.entries(e))n&&t.push(`-e`,`${n}=${r}`);return t}parseDockerRunArgs(e){return e?e.split(/\s+/).map(e=>e.trim()).filter(e=>e.length>0):[]}isDockerVmAlias(e){if(!e)return!1;let t=e.trim().toLowerCase();return Bt.has(t)}async resolveLaunchCommand(e){let{command:t,launchMode:n,vmDockerBrowser:r}=e;if(n===`vm`&&r===`chrome-for-testing`){if(!t)return this.findChromeExecutable();if(this.isChromeTestingAlias(t)||this.isGenericChromeAlias(t))return this.chromeForTesting.getExecutablePath()}return t||this.findChromeExecutable()}isChromeTestingAlias(e){let t=e.trim().toLowerCase();return t===`chrome-testing`||t===`chrome-for-testing`||t===`google-chrome-for-testing`||t.endsWith(`google chrome for testing`)||t.endsWith(`google-chrome-for-testing`)||t.endsWith(`chrome-for-testing.exe`)}isGenericChromeAlias(e){let t=e.trim().toLowerCase(),n=x.basename(t);return n===`chrome`||n===`google-chrome`||n===`googlechrome`}resolveExtensionPath(e){if(e&&O.existsSync(x.join(e,`manifest.json`)))return e;let t=x.dirname(ye(import.meta.url)),n=this.findPackageRoot(t),r=[n?x.resolve(n,`dist/extension`):``,x.resolve(t,`../dist/extension`),x.resolve(t,`../../dist/extension`)].filter(Boolean);for(let e of r)if(O.existsSync(x.join(e,`manifest.json`)))return e;let i=x.resolve(process.cwd(),`dist/extension`);if(O.existsSync(x.join(i,`manifest.json`)))return i;throw Error(`Chrome extension not found. Build the extension first: pnpm build:extension`)}findPackageRoot(e,t=`@agimon-ai/browse-tool`){let n=e;for(let e=0;e<8;e++){let e=x.join(n,`package.json`);if(O.existsSync(e))try{if(JSON.parse(O.readFileSync(e,`utf8`))?.name===t)return n}catch{}let r=x.dirname(n);if(r===n)break;n=r}return null}getExtensionUserDataDir(e){return x.join(this.profileService.getProfilesDir(),e,`extension-user-data`)}createUserDataDir(e){if(e){let t=this.getExtensionUserDataDir(e);return O.mkdirSync(t,{recursive:!0}),this.cleanupExtensionSessionRestoreArtifacts(t),t}return O.mkdtempSync(x.join(N.tmpdir(),`extension-chrome-`))}cleanupExtensionSessionRestoreArtifacts(e){for(let t of this.getSessionRestoreProfileDirs(e))for(let e of qt)try{O.rmSync(x.join(t,e),{recursive:!0,force:!0})}catch{}}getSessionRestoreProfileDirs(e){let t=[e],n=(()=>{try{return O.readdirSync(e,{withFileTypes:!0})}catch{return null}})();if(!n)return t;for(let r of n)r.isDirectory()&&(Jt.has(r.name)||r.name.startsWith(`Profile `))&&t.push(x.join(e,r.name));return t}async findChromeExecutable(){try{return await this.chromeForTesting.getExecutablePath()}catch(e){if(at())throw e}let e=process.platform,t={darwin:[`/Applications/Google Chrome.app/Contents/MacOS/Google Chrome`,`/Applications/Google Chrome Canary.app/Contents/MacOS/Google Chrome Canary`,`${N.homedir()}/Applications/Google Chrome.app/Contents/MacOS/Google Chrome`],linux:[`/usr/bin/google-chrome`,`/usr/bin/google-chrome-stable`,`/usr/bin/chromium`,`/usr/bin/chromium-browser`,`/snap/bin/chromium`],win32:[`${process.env.LOCALAPPDATA}\\Google\\Chrome\\Application\\chrome.exe`,`${process.env.PROGRAMFILES}\\Google\\Chrome\\Application\\chrome.exe`,`${process.env[`PROGRAMFILES(X86)`]}\\Google\\Chrome\\Application\\chrome.exe`]};for(let n of t[e]??[])if(O.existsSync(n))return n;throw Error(`Chrome not found. Install Google Chrome or run with Chrome for Testing.`)}async newPage(e,t){let n=this.browsers.get(e);if(!n)throw Error(`Browser "${e}" not found`);if(n.mode===`extension`||n.mode===`vm`)throw Error(`Cannot create new page in extension mode browser. Use extension APIs directly.`);if(!n.context||!n.browser)throw Error(`Browser "${e}" has no Playwright context`);await this.enforceMaxPageLimit(n);let r=await n.context.newPage(),i=await this.pageRegistry.register({page:r,browser:n.browser,context:n.context,browserId:e,profileName:n.profileName,onClose:r=>{if(n.pageIds.delete(r),n.currentPageId===r){let e=Array.from(n.pageIds);n.currentPageId=e.length>0?e[0]:null}t?.(r),n.pageIds.size===0&&this.closeBrowser(e).catch(()=>{})}});return this.pageMonitor?.startMonitoring(i,r),n.pageIds.add(i),{pageId:i,page:r}}getBrowser(e){return this.browsers.get(e)}getDefaultBrowser(){let e=this.listBrowsers();for(let t of e){if(this.isBrowserAlive(t))return t;this.cleanupStaleBrowser(t.id,t.profileName)}}async getOrCreateDefaultBrowser(e){return this.getDefaultBrowser()||(await this.launch(e)).browserInstance}setCurrentPage(e,t){let n=this.browsers.get(e);if(!n)throw Error(`Browser "${e}" not found`);if(!n.pageIds.has(t))throw Error(`Page "${t}" does not belong to browser "${e}"`);n.extensionRecording?.active&&n.extensionRecording.pageId!==t&&console.warn(`[BrowserService] Extension recording for browser "${e}" remains pinned to initial page "${n.extensionRecording.pageId}"`),n.currentPageId=t}async closeBrowser(e){let t=this.browsers.get(e);if(t){let n,r=t.extensionRecording,i=r?.outputPath,a=r?.active===!0,o=!1;if(r&&this.closingExtensionRecordings.set(e,r),a)try{if(n=(await this.attemptExtensionRecordingFinalization(e)).error,n&&r&&this.shouldCloseRecordedExtensionPageFirst(t)&&(await this.closeRecordedExtensionPage(t,r),i)){if(!await this.waitForExtensionRecordingArtifact(i))throw Error(`Extension recording for browser "${e}" was not persisted after the recorded tab closed`);r.active=!1,n=void 0}}catch(e){n=e instanceof Error?e:Error(String(e))}else try{await this.finalizeExtensionRecording(e)}catch(e){n=e instanceof Error?e:Error(String(e))}if(t.browser)await t.browser.close();else if(t.context)if(t.pid)try{await Lt(t.pid)}catch{try{let e=t.context.pages();await Promise.all(e.map(e=>e.close().catch(e=>{}))),await t.context.close()}catch{}}else{let e=t.context.pages();await Promise.all(e.map(e=>e.close().catch(e=>{}))),await t.context.close()}else if(t.pid)try{await Lt(t.pid)}catch{}if(i&&r&&!O.existsSync(i)){let t=await this.waitForExtensionRecordingArtifactOrChunks(r),a=t.artifactReady;!a&&t.chunkReady&&(a=await this.persistExtensionRecordingArtifact(e)||O.existsSync(i)),a?n=void 0:r&&(o=!0)}if(t.profileName&&this.profileToBrowserId.delete(t.profileName),this.browsers.delete(e),o?this.finalizeClosingExtensionRecording(e):this.closingExtensionRecordings.delete(e),n)throw n}}async closeAll(){let e=Array.from(this.browsers.keys());await Promise.all(e.map(e=>this.closeBrowser(e).catch(()=>{}))),this.browsers.clear(),this.profileToBrowserId.clear(),this.pageRegistry.clear()}listBrowsers(){return Array.from(this.browsers.values())}touchBrowser(e){let t=this.browsers.get(e);t&&(t.lastAccessedAt=new Date)}recordBrowserActivity(e,t){let n=this.browsers.get(e);return n?(n.lastAccessedAt=new Date,t&&this.pageRegistry.touchPage(t),!0):!1}registerExternalBrowser(e){let t=`browser-${++this.browserIdCounter}`,n=new Date,r={id:t,mode:`playwright`,browser:e,context:void 0,pageIds:new Set,currentPageId:null,createdAt:n,lastAccessedAt:n,specOrigin:!0};return this.browsers.set(t,r),e.on(`disconnected`,()=>{this.cleanupStaleBrowser(t)}),t}registerExtensionBrowser(){let e=`browser-${++this.browserIdCounter}`,t=new Date,n={id:e,mode:`extension`,browser:void 0,context:void 0,pageIds:new Set,currentPageId:null,createdAt:t,lastAccessedAt:t};return this.browsers.set(e,n),e}registerExtensionBrowserWithId(e){if(this.browsers.has(e))return;let t=new Date,n={id:e,mode:`extension`,browser:void 0,context:void 0,pageIds:new Set,currentPageId:null,createdAt:t,lastAccessedAt:t};this.browsers.set(e,n)}async ensureExtensionRecordingActive(e,t){let n=this.browsers.get(e),r=n?.extensionRecording;!n||!r||r.pageId!==t||r.active||(r.startPromise||=this.startExtensionRecording(n,r).finally(()=>{n.extensionRecording===r&&(n.extensionRecording.startPromise=void 0)}),await r.startPromise)}async startPageRecording(e,t,n){let r=this.browsers.get(e);if(!r)throw Error(`Browser "${e}" not found`);if(r.mode!==`extension`&&r.mode!==`vm`)throw Error(`Browser "${e}" does not support extension recording in "${r.mode}" mode`);let i=r.extensionRecording;if(i&&i.pageId!==t){if(i.active)throw Error(`Browser "${e}" is already recording page "${i.pageId}". Stop the current recording before starting another one.`);r.extensionRecording=void 0,i=void 0}let a=n?x.resolve(n):i?.outputPath??x.join(N.tmpdir(),`browse-tool-recordings`,`recording-${this.toSafeRecordingId(e)}-${Date.now()}.webm`);if(!i)this.initializeExtensionRecordingAtPath(r,t,a),i=r.extensionRecording;else if(i.outputPath!==a){if(i.active)throw Error(`Browser "${e}" is already recording to "${i.outputPath}". Stop the current recording before changing output path.`);i.outputPath=a,i.chunkPath=`${a}.part`,this.clearExtensionRecordingChunkState(i)}if(!i)throw Error(`Failed to initialize recording for browser "${e}"`);return await this.ensureExtensionRecordingActive(e,t),{outputPath:i.outputPath}}async finalizeExtensionRecording(e,t){let n=this.browsers.get(e),r=n?.extensionRecording;if(!n||!r||t&&r.pageId!==t||!r.active)return;if(!this.extensionTaskQueue)throw Error(`Extension recording task queue is unavailable for browser "${e}"`);let i=st();Kt&&console.log(`[ExtensionRecordingDebug] queue stop browser=${e} page=${r.pageId} active=${r.active} cleanupUi=${i}`);let a;try{a=await this.extensionTaskQueue.queueTask(`browser_stop_recording`,{pageId:r.pageId,cleanupUi:i},H,e)}finally{r.active=!1}Kt&&console.log(`[ExtensionRecordingDebug] stop result browser=${e} success=${a?.success===!0} isError=${a?.result?.isError===!0}`);let o;try{o=this.parseExtensionRecordingPayload(a)}catch(t){if(await this.persistExtensionRecordingArtifact(e))return;throw t}if(o.videoBase64){O.mkdirSync(x.dirname(r.outputPath),{recursive:!0}),O.writeFileSync(r.outputPath,Buffer.from(o.videoBase64,`base64`)),this.clearExtensionRecordingChunkState(r),r.active=!1;return}if(!await this.persistExtensionRecordingArtifact(e))throw Error(`Extension recording for browser "${e}" stopped without video data`)}async stopPageRecording(e,t,n={}){let r=this.browsers.get(e)?.extensionRecording??this.closingExtensionRecordings.get(e);if(!r)throw Error(`Browser "${e}" has no recording session`);if(t&&r.pageId!==t)throw Error(`Page "${t}" does not match the active recording for browser "${e}"`);this.closingExtensionRecordings.set(e,r);let i;try{i=(await this.attemptExtensionRecordingFinalization(e)).error}catch(e){i=e instanceof Error?e:Error(String(e))}if(!O.existsSync(r.outputPath)){let t=await this.waitForExtensionRecordingArtifactOrChunks(r),n=t.artifactReady;if(!n&&t.chunkReady&&(n=await this.persistExtensionRecordingArtifact(e)||O.existsSync(r.outputPath)),!n&&i)throw i}if(!O.existsSync(r.outputPath))throw Error(`Recording artifact for browser "${e}" was not written to "${r.outputPath}"`);let a=O.readFileSync(r.outputPath);return this.closingExtensionRecordings.delete(e),{outputPath:r.outputPath,fileSizeBytes:a.byteLength,...n.includeBase64?{videoBase64:a.toString(`base64`)}:{}}}async persistExtensionRecordingArtifact(e,t){let n=this.getExtensionRecordingSession(e);if(!n)return!1;if(!t&&this.hasPersistedExtensionRecordingArtifact(n))return n.active=!1,this.closingExtensionRecordings.delete(e),!0;if(O.mkdirSync(x.dirname(n.outputPath),{recursive:!0}),t)O.writeFileSync(n.outputPath,Buffer.from(t,`base64`)),this.clearExtensionRecordingChunkState(n);else if(!this.persistExtensionRecordingChunksToArtifact(n))return!1;return n.active=!1,this.closingExtensionRecordings.delete(e),!0}async persistExtensionRecordingChunk(e,t){let n=this.getExtensionRecordingSession(e);if(!n||!t)return null;let r=Buffer.from(t,`base64`);return r.length===0?null:(O.mkdirSync(x.dirname(n.chunkPath),{recursive:!0}),O.appendFileSync(n.chunkPath,r),n.chunkCount+=1,n.chunkBytes+=r.length,{chunkBytes:r.length,totalBytes:n.chunkBytes,chunkCount:n.chunkCount})}async enforceMaxPageLimit(e){if(e.pageIds.size<U.MAX_PAGES_PER_BROWSER)return;let t=this.pageRegistry.findByBrowser(e.id);t.sort((e,t)=>e.createdAt.getTime()-t.createdAt.getTime());let n=t[0];n&&(n.page?await n.page.close():(this.pageRegistry.remove(n.id),e.pageIds.delete(n.id)))}async enforceMaxBrowserLimit(){if(this.browsers.size>=U.MAX_ACTIVE_BROWSERS){let e=Array.from(this.browsers.values());e.sort((e,t)=>e.createdAt.getTime()-t.createdAt.getTime());let t=e[0];t&&await this.closeBrowser(t.id)}}isBrowserAlive(e){try{if(e.mode===`playwright`)return e.browser?.isConnected()??!1;if(e.mode===`extension`||e.mode===`vm`){if(!e.pid||!this.isProcessRunning(e.pid))return!1;if(!e.context)return!0;try{return e.context.pages(),!0}catch{return!1}}return!1}catch{return!1}}isProcessRunning(e){try{return process.kill(e,0),!0}catch(e){return e.code===`EPERM`}}captureProcessOutput(){let e=[],t=0,n=(n,r)=>{if(typeof r!=`string`&&!Buffer.isBuffer(r))return;let i=typeof r==`string`?r:r.toString(`utf8`);if(!i)return;let a=32768-t;if(a<=0)return;let o=Buffer.from(i,`utf8`).subarray(0,a).toString(`utf8`);t+=Buffer.byteLength(o,`utf8`),e.push(`[${n}] ${o}`)};return{attach:e=>{e.stdout?.on(`data`,e=>n(`stdout`,e)),e.stderr?.on(`data`,e=>n(`stderr`,e))},snapshot:()=>e.join(``)}}async startExtensionBrowser(e){let{browserId:t,launchMode:n,profileName:r,vmRuntime:i,spawnArgs:a,recordVideo:o,startupDelayMs:s,launchUrl:c}=e,l=this.captureProcessOutput(),u=F(i.command,a,{detached:!1,stdio:[`ignore`,`pipe`,`pipe`],env:i.kind===`docker`?process.env:{...process.env,DISPLAY:process.env.DISPLAY}});if(!u.pid)throw Error(`Failed to start Chrome process`);l.attach(u);let d=new Date,f={id:t,mode:n,browser:void 0,context:void 0,profileName:r,pageIds:new Set,currentPageId:null,pid:u.pid,createdAt:d,lastAccessedAt:d};u.on(`exit`,()=>this.cleanupStaleBrowser(t,r)),u.on(`error`,()=>this.cleanupStaleBrowser(t,r)),this.browsers.set(t,f),r&&this.profileToBrowserId.set(r,t);let p=this.pageRegistry.registerExtensionPage(t);f.pageIds.add(p),f.currentPageId=p;try{if(await this.waitForExtensionReady(t,u,l,{vmRuntimeKind:i.kind,launchUrl:c}),o){Kt&&console.log(`[ExtensionRecordingDebug] initialize recording browser=${f.id} page=${p} dir=${o.dir}`),this.initializeExtensionRecording(f,p,o);let e=f.extensionRecording;e&&this.startExtensionRecording(f,e).catch(e=>{console.warn(`[BrowserService] Failed to eagerly start extension recording for browser "${f.id}" on page "${p}":`,e)})}}catch(e){throw await this.closeBrowser(t).catch(()=>{}),e}return s&&s>0&&await this.delay(s),{browserInstance:f,pageId:p,process:u}}async waitForExtensionReady(e,t,n,r){if(!this.webSocketHub&&!this.extensionSessionRegistry)return;let i=null,a=null,o;t.once(`exit`,(e,t)=>{i=e,a=t}),t.once(`error`,e=>{o=e});let s=null,c=!1,l=0,u=0,d=null,f=!1,p=Date.now(),m=this.getExtensionStartupTimeoutMs(r.vmRuntimeKind),h=this.getExtensionStartupGraceTimeoutMs(r.vmRuntimeKind),g=this.getExtensionReadyStabilityMs(),_=this.getExtensionStartupPollIntervalMs(),v=p+m;for(;;){let t=Date.now();if(t>=v){if(!f&&o===void 0&&i===null&&a===null){f=!0,v=t+h;continue}break}let n=this.getExtensionReadyState(e);if(n.connected){if(c=!0,n.explicit||(s??=t,d??=t,u=Math.max(u,t-s),t-s>=g))return}else s!==null&&(l+=1,u=Math.max(u,t-s)),s=null;if(o||i!==null||a!==null)break;await new Promise(e=>setTimeout(e,_))}let y=this.getExtensionReadyState(e);if(y.connected&&(y.explicit||s!==null&&Date.now()-s>=g))return;let b=n.snapshot(),x=this.writeLaunchFailureLog(e,b),S=[`Chrome extension browser "${e}" failed to become ready within ${m+(f?h:0)}ms.`,`Ready state requires a stable session for ${g}ms.`,`Observed connection: ${c?`yes`:`no`}.`,c&&d!==null?`First connection observed after ${Math.max(0,d-p)}ms.`:void 0,`Longest stable connection: ${u}ms.`,l>0?`Connection flaps: ${l}.`:void 0,f?`Startup grace applied: ${h}ms.`:void 0,r.launchUrl?`Launch URL: ${r.launchUrl}.`:void 0,o?`Error: ${o.message}`:void 0,i===null?void 0:`Exit code: ${i}`,a?`Signal: ${a}`:void 0,x?`Launch log: ${x}`:void 0].filter(Boolean).join(` `);throw Error(S)}getExtensionStartupTimeoutMs(e){return e===`docker`?12e3:5e3}getExtensionStartupGraceTimeoutMs(e){return e===`docker`?8e3:5e3}getExtensionReadyStabilityMs(){return 1500}getExtensionStartupPollIntervalMs(){return 100}isRetryableExtensionLaunchError(e){return/failed to become ready/i.test(e.message)||/Failed to start Chrome process/i.test(e.message)}shouldSerializeChromeForTestingLaunch(e){return process.platform===`darwin`&&e.kind===`host`&&this.isChromeTestingAlias(e.command)}async runSerializedChromeForTestingLaunch(e){let t,n=U.chromeForTestingLaunchQueue;U.chromeForTestingLaunchQueue=new Promise(e=>{t=e}),await n;try{return await e()}finally{t?.()}}async delay(e){await new Promise(t=>setTimeout(t,e))}getExtensionReadyState(e){return this.extensionSessionRegistry?{connected:!!this.extensionSessionRegistry.getSessionByBrowserId(e),explicit:!0}:{connected:!!this.webSocketHub?.hasConnection(e),explicit:!1}}hasExtensionConnection(e){return this.getExtensionReadyState(e).connected}canReuseExtensionBrowser(e){return e.mode!==`extension`&&e.mode!==`vm`||!e.pid?!1:this.isProcessRunning(e.pid)&&this.hasExtensionConnection(e.id)}getTrackedExtensionPageId(e){let t=e.currentPageId&&this.pageRegistry.get(e.currentPageId)?e.currentPageId:Array.from(e.pageIds).find(e=>this.pageRegistry.get(e));if(t)return e.currentPageId=t,t;let n=this.pageRegistry.registerExtensionPage(e.id);return e.pageIds.add(n),e.currentPageId=n,n}writeLaunchFailureLog(e,t){if(t.trim())try{let n=x.join(N.tmpdir(),`${e}-launch.log`);return O.writeFileSync(n,t,`utf8`),n}catch{return}}async prepareDockerRetryUserDataDir(e){await this.delay(250);try{let t=await this.lockManager.cleanupOrphanedLocks(e);return this.cleanupExtensionSingletonArtifacts(e),`${t?`cleaned`:`checked`} ${e}`}catch(t){let n=t instanceof Error?t.message:String(t);return this.cleanupExtensionSingletonArtifacts(e),`lock-manager cleanup failed (${n}); singleton artifacts removed directly from ${e}`}}cleanupExtensionSingletonArtifacts(e){let t=[x.join(e,`SingletonLock`),x.join(e,`SingletonCookie`),x.join(e,`SingletonSocket`),x.join(e,`lockfile`),x.join(e,`Default`,`LOCK`)];for(let e of t)try{O.rmSync(e,{force:!0})}catch{}}cleanupStaleBrowser(e,t){if(t){let e=this.getExtensionUserDataDir(t);this.cleanupExtensionSingletonArtifacts(e),this.lockManager.cleanupOrphanedLocks(e).catch(()=>{})}this.browsers.delete(e),t&&this.profileToBrowserId.delete(t)}initializeExtensionRecording(e,t,n){let r=x.join(n.dir,`recording-${this.toSafeRecordingId(e.id)}-${e.createdAt.getTime()}.webm`);this.initializeExtensionRecordingAtPath(e,t,r)}initializeExtensionRecordingAtPath(e,t,n){e.extensionRecording={pageId:t,outputPath:n,chunkPath:`${n}.part`,active:!1,chunkCount:0,chunkBytes:0}}async startExtensionRecording(e,t){if(!this.extensionTaskQueue)throw Error(`Extension recording task queue is unavailable for browser "${e.id}"`);Kt&&console.log(`[ExtensionRecordingDebug] queue start browser=${e.id} page=${t.pageId} active=${t.active}`);let n=await this.extensionTaskQueue.queueTask(`browser_start_recording`,{pageId:t.pageId},H,e.id);if(Kt&&console.log(`[ExtensionRecordingDebug] start result browser=${e.id} success=${n?.success===!0} isError=${n?.result?.isError===!0}`),!n.success||n.result?.isError){let r=this.getExtensionTaskError(n);if(this.isExtensionRecordingAlreadyActiveError(r)){t.active=!0;return}throw Error(r??`Failed to start extension recording for "${e.id}"`)}t.active=!0}isExtensionRecordingAlreadyActiveError(e){return e?/active stream/i.test(e)||/recording already in progress/i.test(e):!1}async attemptExtensionRecordingFinalization(e){let t=this.finalizeExtensionRecording(e).then(()=>({completed:!0})).catch(e=>({completed:!0,error:e instanceof Error?e:Error(String(e))})),n=this.delay(1500).then(()=>({completed:!1}));return Promise.race([t,n])}async waitForExtensionRecordingArtifact(e){let t=Date.now();for(;Date.now()-t<Gt;){try{let t=O.statSync(e);if(t.isFile()&&t.size>0)return!0}catch{}await this.delay(100)}return!1}async waitForExtensionRecordingArtifactOrChunks(e){let t=Date.now();for(;Date.now()-t<Gt;){try{let t=O.statSync(e.outputPath);if(t.isFile()&&t.size>0)return{artifactReady:!0,chunkReady:!1}}catch{}try{let t=O.statSync(e.chunkPath);if(t.isFile()&&t.size>0)return{artifactReady:!1,chunkReady:!0}}catch{}await this.delay(100)}return{artifactReady:!1,chunkReady:!1}}getExtensionRecordingSession(e){return this.browsers.get(e)?.extensionRecording??this.closingExtensionRecordings.get(e)}async finalizeClosingExtensionRecording(e){let t=this.closingExtensionRecordings.get(e);if(t)try{let n=await this.waitForExtensionRecordingArtifactOrChunks(t);!n.artifactReady&&n.chunkReady&&await this.persistExtensionRecordingArtifact(e)}finally{this.closingExtensionRecordings.delete(e)}}shouldCloseRecordedExtensionPageFirst(e){return(e.mode===`extension`||e.mode===`vm`)&&!!e.extensionRecording&&!!this.extensionTaskQueue}async closeRecordedExtensionPage(e,t){if(!this.extensionTaskQueue)throw Error(`Extension recording task queue is unavailable for browser "${e.id}"`);if(e.pageIds.size<=1){let t=await this.extensionTaskQueue.queueTask(`browser_new_page`,{url:`about:blank`,setAsCurrent:!1},H,e.id);if(!t.success||t.result?.isError)throw Error(this.getExtensionTaskError(t)??`Failed to create a keepalive tab for browser "${e.id}" before closing the recorded page`)}let n=await this.extensionTaskQueue.queueTask(`browser_close_page`,{pageId:t.pageId},H,e.id);if(!n.success||n.result?.isError)throw Error(this.getExtensionTaskError(n)??`Failed to close the recorded page "${t.pageId}" for browser "${e.id}"`)}parseExtensionRecordingPayload(e){if(!e.success||e.result?.isError)throw Error(this.getExtensionTaskError(e)??`Extension recording task failed`);let t=e.result?.content[0],n=t?.type===`text`?t.text:void 0;if(!n)return{};try{return JSON.parse(n)}catch{throw Error(`Extension recording task returned invalid JSON payload`)}}getExtensionTaskError(e){if(e.error)return e.error;let t=e.result?.content[0];if(t?.type===`text`)return t.text}toSafeRecordingId(e){return e.replace(/[^a-zA-Z0-9_-]+/g,`-`)}persistExtensionRecordingChunksToArtifact(e){try{let t=O.statSync(e.chunkPath);if(!t.isFile()||t.size<=0)return!1}catch{return!1}if(O.existsSync(e.outputPath)){let t=O.statSync(e.outputPath);if(t.isFile()&&t.size>0)return this.clearExtensionRecordingChunkState(e),!0}return O.renameSync(e.chunkPath,e.outputPath),e.chunkCount=0,e.chunkBytes=0,!0}hasPersistedExtensionRecordingArtifact(e){try{let t=O.statSync(e.outputPath);return t.isFile()&&t.size>0}catch{return!1}}clearExtensionRecordingChunkState(e){e.chunkCount=0,e.chunkBytes=0;try{O.existsSync(e.chunkPath)&&O.rmSync(e.chunkPath,{force:!0})}catch{}}getBrowserType(e){switch(e){case`chromium`:return xe;case`firefox`:return Se;case`webkit`:return Ce}}buildContextOptions(e,t,n){let r={};return t&&(r.baseURL=t),n&&(r.recordVideo=n),e?(e.viewport&&(r.viewport=e.viewport),e.userAgent&&(r.userAgent=e.userAgent),e.locale&&(r.locale=e.locale),e.timezone&&(r.timezoneId=e.timezone),e.geolocation&&(r.geolocation=e.geolocation),e.permissions&&(r.permissions=e.permissions),e.colorScheme&&(r.colorScheme=e.colorScheme),r):r}};Yt=U=V([v(),B(0,_(R.ProfileService)),B(1,_(R.PageRegistry)),B(2,_(R.BrowserLockManager)),B(3,_(R.ChromeForTestingService)),B(4,_(R.PageMonitorService)),B(4,y()),B(5,_(R.WebSocketHub)),B(5,y()),B(6,_(R.ExtensionSessionRegistry)),B(6,y()),B(7,_(R.ExtensionTaskQueue)),B(7,y()),z(`design:paramtypes`,[Object,Object,Object,Object,Object,Object,Object,Object])],Yt);function Xt(e){return e.trim().toLowerCase().replace(/[^a-z0-9]+/g,`-`).replace(/^-+|-+$/g,``)||`snippet`}function Zt(e){return e.replace(/\r\n/g,`
|
|
6
|
+
`).split(`
|
|
7
|
+
`).map(e=>e.length>0?` ${e}`:e).join(`
|
|
8
|
+
`)}function Qt(e){return[`export const name = ${JSON.stringify(e.name)};`,`export const description = ${JSON.stringify(e.description)};`,`export const run = async ({ page, context, browser }) => {`,Zt(e.code),`};`,``].join(`
|
|
9
|
+
`)}async function $t(e){let t=c(await j(e,`utf8`),{mode:`strip`});return await import(`data:text/javascript;base64,${Buffer.from(t,`utf8`).toString(`base64`)}`)}var en=class{constructor(e=process.env.BROWSE_TOOL_SNIPPETS_DIR){this.snippetsDir=e}getDirectory(){return this.snippetsDir?x.resolve(this.snippetsDir):void 0}async listSnippets(){let e=this.requireDirectory();await A(e,{recursive:!0});let t=await ce(e,{withFileTypes:!0}),n=[];for(let r of t){if(!r.isFile()||!r.name.endsWith(`.ts`))continue;let t=r.name,i=await $t(x.join(e,r.name)),a=typeof i.name==`string`&&i.name.trim().length>0?i.name:r.name.replace(/\.ts$/,``),o=typeof i.description==`string`?i.description:``;n.push({name:a,description:o,snippetPath:t})}return n.sort((e,t)=>e.name.localeCompare(t.name)),n}async loadSnippet(e){let t=await this.resolveSnippetPath(e),n=await $t(t.filePath);if(typeof n.run!=`function`)throw Error(`Snippet "${t.snippetPath}" must export a "run" function`);return{name:typeof n.name==`string`&&n.name.trim().length>0?n.name:t.snippetPath.replace(/\.ts$/,``),description:typeof n.description==`string`?n.description:``,snippetPath:t.snippetPath,run:n.run}}async saveSnippet(e){let t=this.requireDirectory();await A(t,{recursive:!0});let n=`${Xt(e.name)}.ts`;return await M(x.join(t,n),Qt(e),`utf8`),{name:e.name,description:e.description,snippetPath:n}}requireDirectory(){let e=this.getDirectory();if(!e)throw Error(`Snippet storage is not configured. Start browse-tool with --snippets-dir.`);return e}async resolveSnippetPath(e){let t=this.requireDirectory(),n=e.endsWith(`.ts`)?e:`${e}.ts`,r=x.resolve(t,n);if(!r.startsWith(`${t}${x.sep}`)&&r!==x.join(t,n))throw Error(`Snippet path must stay within the configured snippets directory`);if(!(await de(r).catch(()=>null))?.isFile())throw Error(`Snippet "${e}" not found`);return{filePath:r,snippetPath:x.relative(t,r)}}};const W=I.object({uid:I.string().optional().describe(`Accessibility snapshot UID reference`),role:I.string().optional().describe(`ARIA role (e.g., "button", "textbox")`),name:I.string().optional().describe(`Accessible name used with the role selector`),label:I.string().optional().describe(`Associated label text`),placeholder:I.string().optional().describe(`Placeholder text`),testId:I.string().optional().describe(`data-testid value`),text:I.string().optional().describe(`Visible text content`),exact:I.boolean().optional().default(!1).describe(`Use exact matching for semantic selectors`),selector:I.string().optional().describe(`CSS selector`),xpath:I.string().optional().describe(`XPath expression`),frame:I.string().optional().describe(`Frame selector`)});let tn=class{async locate(e,t){let n=await this.getFrameContext(e,t.frame),r=this.buildLocator(n,t);if(await r.count()===0)throw Error(`Element not found: ${this.describeSelector(t)}`);return r.first()}async locateAll(e,t){let n=await this.getFrameContext(e,t.frame);return this.buildLocator(n,t)}async waitForElement(e,t,n={}){let r=await this.getFrameContext(e,t.frame),i=this.buildLocator(r,t);return await i.waitFor({timeout:n.timeout??18e4,state:n.state??`visible`}),i.first()}async getFrameContext(e,t){return t?e.frameLocator(t):e}buildLocator(e,t){let n=t.exact??!1;if(t.uid)return e.locator(`[data-pw-proxy="${t.uid}"], [data-browse-tool-uid="${t.uid}"], [data-uid="${t.uid}"]`);if(t.role){let r=t.role;return t.name?e.getByRole(r,{name:t.name,exact:n}):e.getByRole(r)}if(t.label)return e.getByLabel(t.label,{exact:n});if(t.placeholder)return e.getByPlaceholder(t.placeholder,{exact:n});if(t.testId)return e.getByTestId(t.testId);if(t.text)return e.getByText(t.text,{exact:n});if(t.selector)return e.locator(t.selector);if(t.xpath)return e.locator(`xpath=${t.xpath}`);throw Error(`No selector provided. Specify one of: uid, role, label, placeholder, text, selector, or xpath.`)}describeSelector(e){let t=[];return e.uid&&t.push(`uid="${e.uid}"`),e.role&&t.push(`role="${e.role}"`),e.name&&t.push(`name="${e.name}"`),e.label&&t.push(`label="${e.label}"`),e.placeholder&&t.push(`placeholder="${e.placeholder}"`),e.testId&&t.push(`testId="${e.testId}"`),e.text&&t.push(`text="${e.text}"`),e.exact&&t.push(`exact=true`),e.selector&&t.push(`selector="${e.selector}"`),e.xpath&&t.push(`xpath="${e.xpath}"`),e.frame&&t.push(`frame="${e.frame}"`),t.length>0?t.join(`, `):`empty selector`}};tn=V([v()],tn);const nn=process.env.BROWSE_TOOL_DEBUG_CUSTOM_TOOLS===`1`;function rn(e){return JSON.stringify(e,(e,t)=>typeof t==`string`&&t.length>240?`${t.slice(0,240)}...<trimmed>`:t)}function an(e,t){if(nn){if(t){console.error(`[ExtensionPageProxy] ${e}`,t);return}console.error(`[ExtensionPageProxy] ${e}`)}}function on(e){return typeof e==`object`&&e&&`value`in e?e.value:e}let sn=class{[o]=!0;currentUrl=`about:blank`;defaultTimeoutMs=H;_pageId;_browserId;constructor(e){this.taskQueue=e}setTarget(e,t){this._pageId=e,this._browserId=t,this.currentUrl=`about:blank`,this.currentUrlAsync().catch(()=>{})}get pageId(){return this._pageId}get browserId(){return this._browserId}async goto(e,t){let n=await this.executeTask(`browser_navigate`,{url:e,waitUntil:t?.waitUntil,timeout:t?.timeout});if(!n.success)throw Error(n.error??`Navigation failed`);this.currentUrl=e}async click(e,t){let n=await this.executeTask(`browser_click`,{selector:e,clickCount:t?.clickCount,delay:t?.delay,button:t?.button,modifiers:t?.modifiers});if(!n.success)throw Error(n.error??`Click failed on ${e}`)}async fill(e,t,n){let r=await this.executeTask(`browser_fill`,{selector:e,text:t,force:n?.force});if(!r.success)throw Error(r.error??`Fill failed on ${e}`)}async type(e,t,n){let r=await this.executeTask(`browser_type`,{selector:e,text:t,delay:n?.delay});if(!r.success)throw Error(r.error??`Type failed on ${e}`)}async press(e){let t=await this.executeTask(`browser_press_key`,{key:e});if(!t.success)throw Error(t.error??`Press key failed: ${e}`)}async hover(e){let t=await this.executeTask(`browser_hover`,{selector:e});if(!t.success)throw Error(t.error??`Hover failed on ${e}`)}async selectOption(e,t){let n=await this.executeTask(`browser_select`,{selector:e,values:Array.isArray(t)?t:[t]});if(!n.success)throw Error(n.error??`Select option failed on ${e}`)}async waitForSelector(e,t){let n=await this.executeTask(`browser_wait_for`,{selector:e,state:t?.state??`visible`,timeout:t?.timeout});if(!n.success)throw Error(n.error??`Wait for selector failed: ${e}`)}async waitForTimeout(e){let t=await this.executeTask(`browser_wait_for`,{timeout:e});if(!t.success)throw Error(t.error??`Wait for timeout failed`)}async screenshot(e){let t=await this.executeTask(`browser_screenshot`,{path:e?.path,fullPage:e?.fullPage,type:e?.type});if(!t.success)throw Error(t.error??`Screenshot failed`);return t.result?.path??``}async content(){let e=await this.executeTask(`browser_evaluate_script`,{script:`document.documentElement.outerHTML`});if(!e.success)throw Error(e.error??`Get content failed`);return on(e.result)??``}async title(){let e=await this.executeTask(`browser_evaluate_script`,{script:`document.title`});if(!e.success)throw Error(e.error??`Get title failed`);return on(e.result)??``}async textContent(e){return this.locator(e).textContent()}async innerText(e){return this.locator(e).innerText()}async inputValue(e){return this.locator(e).inputValue()}url(){return this.currentUrl}async evaluate(e,...t){let n=typeof e==`function`?`(${e.toString()})(${t.map(e=>JSON.stringify(e)).join(`,`)})`:e,r=await this.executeTask(`browser_evaluate_script`,{script:n});if(!r.success)throw Error(r.error??`Evaluate failed`);return on(r.result)}async getSnapshot(){let e=await this.executeTask(`browser_snapshot`,{});if(!e.success)throw Error(e.error??`Get snapshot failed`);let t=on(e.result);return typeof t==`string`?t:t==null?``:JSON.stringify(t,null,2)}getByRole(e,t){return new s(this,[{type:`role`,role:e,options:t}])}getByText(e,t){return new s(this,[{type:`text`,text:e,options:t}])}getByLabel(e,t){return new s(this,[{type:`label`,text:e,options:t}])}getByPlaceholder(e,t){return new s(this,[{type:`placeholder`,text:e,options:t}])}getByTestId(e){return new s(this,[{type:`testId`,testId:e}])}locator(e){return new s(this,[{type:`css`,selector:e}])}context(){return{clearCookies:async()=>{await this.clearStorageState()}}}async reload(e){let t=await this.executeTask(`browser_reload`,{waitUntil:e?.waitUntil,timeout:e?.timeout});if(!t.success)throw Error(t.error??`Reload failed`)}async currentUrlAsync(){let e=await this.evaluate(`window.location.href`);return e&&(this.currentUrl=e),this.currentUrl}async waitForResponse(e,t){let n=typeof e==`string`?{url:e}:e instanceof RegExp?{urlPattern:{source:e.source,flags:e.flags}}:null;if(!n)throw Error(`Extension mode only supports waitForResponse with a string or RegExp target`);let r=await this.executeTask(`browser_wait_for_response`,{...n,timeout:t?.timeout??this.defaultTimeoutMs});if(!r.success)throw Error(r.error??`Wait for response failed`)}async getStorageState(){let e=await this.executeTask(`browser_get_storage_state`,{});if(!e.success)throw Error(e.error??`Get storage state failed`);return e.result??{}}async clearStorageState(){let e=await this.executeTask(`browser_clear_storage_state`,{});if(!e.success)throw Error(e.error??`Clear storage state failed`)}async startRecording(){let e=await this.executeTask(`browser_start_recording`,{});if(!e.success)throw Error(e.error??`Start recording failed`)}async stopRecording(){let e=await this.executeTask(`browser_stop_recording`,{});if(!e.success)throw Error(e.error??`Stop recording failed`);return e.result?.videoBase64??``}async executeTask(e,t){if(!this._pageId||!this._browserId)return{success:!1,error:`Extension page proxy target is not initialized`};let n={...t,pageId:this._pageId};an(`Queueing extension task`,{tool:e,browserId:this._browserId,pageId:this._pageId,args:rn(n)});let r;try{r=await this.taskQueue.queueTask(e,n,this.defaultTimeoutMs,this._browserId)}catch(t){let n=t instanceof Error?t.message:String(t);return an(`Extension task threw before result`,{tool:e,browserId:this._browserId,pageId:this._pageId,error:n}),{success:!1,error:`Extension task "${e}" threw for page "${this._pageId}" in browser "${this._browserId}": ${n}`}}if(!r.success)return an(`Extension task returned unsuccessful result`,{tool:e,browserId:this._browserId,pageId:this._pageId,taskError:r.error??`Task failed`,taskResult:r}),{success:!1,error:r.error??`Extension task "${e}" failed for page "${this._pageId}" in browser "${this._browserId}"`};let i=r.result?.content?.[0];if(i?.type===`text`&&i.text)try{let t=JSON.parse(i.text);return an(`Extension task completed with JSON text result`,{tool:e,browserId:this._browserId,pageId:this._pageId}),{success:!0,result:t}}catch{return an(`Extension task completed with plain text result`,{tool:e,browserId:this._browserId,pageId:this._pageId,textPreview:i.text.slice(0,240)}),{success:!0,result:{value:i.text}}}return an(`Extension task completed with non-text result payload`,{tool:e,browserId:this._browserId,pageId:this._pageId,taskResult:r}),{success:!0,result:r.result}}};sn=V([v(),B(0,_(R.ExtensionTaskQueue)),z(`design:paramtypes`,[Object])],sn);let cn=class{sessions=new Map;browserIdToSessionId=new Map;sessionIdCounter=0;defaultStaleThresholdMs=6e4;register(e){let t=this.browserIdToSessionId.get(e.browserId);t&&(this.sessions.delete(t),this.browserIdToSessionId.delete(e.browserId));let n=`session-${++this.sessionIdCounter}-${Date.now()}`,r=new Date,i={id:n,browserId:e.browserId,tabId:e.tabId,currentUrl:e.url,createdAt:r,lastHeartbeatAt:r,controlMode:`manual`,handoffRequested:!1,metadata:e.metadata};return this.sessions.set(n,i),this.browserIdToSessionId.set(e.browserId,n),i}heartbeat(e){let t=this.sessions.get(e.sessionId);if(t)return t.lastHeartbeatAt=new Date,e.tabId!==void 0&&(t.tabId=e.tabId),e.url!==void 0&&(t.currentUrl=e.url),e.state&&(t.metadata={...t.metadata,...e.state}),t}requestHandoff(e){let t=this.sessions.get(e.sessionId);if(t)return t.handoffRequested=!0,e.pageState&&(t.currentUrl=e.pageState.url,t.metadata={...t.metadata,handoffReason:e.reason,handoffPageState:e.pageState}),t}acknowledgeHandoff(e){let t=this.sessions.get(e);if(!(!t||!t.handoffRequested))return t.handoffRequested=!1,t.controlMode=`ai`,t.activeSpecPath=void 0,t}getSession(e){return this.sessions.get(e)}getSessionByBrowserId(e){let t=this.browserIdToSessionId.get(e);if(t)return this.sessions.get(t)}listSessions(){return Array.from(this.sessions.values())}removeSession(e){let t=this.sessions.get(e);return t?(this.browserIdToSessionId.delete(t.browserId),this.sessions.delete(e),!0):!1}setControlMode(e,t){let n=this.sessions.get(e);if(n)return n.controlMode=t,n}setActiveSpec(e,t){let n=this.sessions.get(e);if(n)return n.activeSpecPath=t,t&&(n.controlMode=`spec`),n}cleanupStaleSessions(e=this.defaultStaleThresholdMs){let t=Date.now(),n=0;for(let[r,i]of this.sessions)t-i.lastHeartbeatAt.getTime()>e&&(this.browserIdToSessionId.delete(i.browserId),this.sessions.delete(r),n++);return n}};cn=V([v()],cn);let ln=class{constructor(e,t,n){this.bundler=e,this.pageProxy=t,this.sessionRegistry=n}async loadSpec(e){let n=await this.bundler.bundle(e),a=process,o=a.__pw_initiator__;try{return t(e),a.__pw_initiator__=void 0,await import(`${n.outputPath}?t=${Date.now()}`),i()}catch(t){throw r(),Error(`Failed to load spec "${e}": ${t instanceof Error?t.message:String(t)}`,{cause:t})}finally{a.__pw_initiator__=o,await n.cleanup()}}async executeSpec(e){let{specPath:t,sessionId:r,testFilter:i,specArgs:a,onHandoff:o}=e,s=Date.now(),c=[],l=0,u=0,d=!1,f=new Set,p=new Set;this.sessionRegistry.setActiveSpec(r,t);try{let e=await this.loadSpec(t),m=i?this.filterTests(e.allTests,i):e.allTests,h={page:this.pageProxy,requestHandoff:async e=>{if(!this.sessionRegistry.getSession(r))throw Error(`Session ${r} not found`);let t=await this.pageProxy.getSnapshot(),n=this.pageProxy.url();try{n=await this.pageProxy.currentUrlAsync()}catch{}this.sessionRegistry.requestHandoff({sessionId:r,reason:e,pageState:{url:n,snapshot:t}}),d=!0,o&&await o(r)},specArgs:a};for(let e of m){let{runtimeUse:t,unsupportedKeys:i,fixtureOverrides:a}=n(e.config.use);i.length>0&&f.add(`test.use() ignores unsupported fixture keys in extension mode: ${i.join(`, `)}`),Object.keys(t).length>0&&f.add(`test.use() runtime browser config is not applied in extension mode: ${Object.keys(t).join(`, `)}`);let o=e.serialScopeId,s=o!==void 0&&p.has(o);if(e.skip||s){c.push({title:e.title,fullTitle:e.fullTitle,passed:!0,error:null,duration:0}),l++;continue}let d=this.sessionRegistry.getSession(r);if(d?.handoffRequested)break;let m=Date.now();try{await e.fn({...h,...a}),c.push({title:e.title,fullTitle:e.fullTitle,passed:!0,error:null,duration:Date.now()-m,handoffTriggered:d?.handoffRequested}),l++}catch(t){let n=t instanceof Error?t.message:String(t);c.push({title:e.title,fullTitle:e.fullTitle,passed:!1,error:n,duration:Date.now()-m}),u++,o!==void 0&&p.add(o)}}return{specPath:t,totalTests:m.length,passed:l,failed:u,testResults:c,success:u===0,duration:Date.now()-s,handoffOccurred:d,sessionId:r,warnings:f.size>0?[...f]:void 0}}catch(e){return{specPath:t,totalTests:0,passed:0,failed:1,testResults:[{title:`Spec Loading`,fullTitle:`Failed to load: ${t}`,passed:!1,error:e instanceof Error?e.message:String(e),duration:Date.now()-s}],success:!1,duration:Date.now()-s,handoffOccurred:!1,sessionId:r,warnings:f.size>0?[...f]:void 0}}finally{this.sessionRegistry.setActiveSpec(r,void 0)}}filterTests(e,t){let n=[...e];if(t.onlyMarked){let e=n.filter(e=>e.only);e.length>0&&(n=e)}if(t.testName&&(n=n.filter(e=>e.title===t.testName)),t.testPattern){let e=new RegExp(t.testPattern,`i`);n=n.filter(t=>e.test(t.fullTitle))}if(t.describeFilter){let e=new RegExp(t.describeFilter,`i`);n=n.filter(t=>e.test(t.fullTitle))}return n}};ln=V([v(),B(0,_(R.SpecBundlerService)),B(1,_(R.ExtensionPageProxy)),B(2,_(R.ExtensionSessionRegistry)),z(`design:paramtypes`,[Object,Object,Object])],ln);const un=`browse-tool-http`,dn=[`pnpm-workspace.yaml`,`nx.json`,`.git`],fn=new Set([`127.0.0.1`,`localhost`,`::1`,`0.0.0.0`]);function pn(e){return e===`1`||e===`true`}function mn(e){return Array.isArray(e)?JSON.stringify(e):typeof e==`string`||typeof e==`number`||typeof e==`boolean`?e:JSON.stringify(e)}function hn(e){if(!e)return;let t={};for(let[n,r]of Object.entries(e))r!=null&&(t[n]=mn(r));return Object.keys(t).length>0?t:void 0}function gn(e){if(!e)return;let t={};for(let n of e.split(`,`)){let[e,...r]=n.split(`=`),i=e?.trim(),a=r.join(`=`).trim();!i||!a||(t[i]=a)}return Object.keys(t).length>0?t:void 0}function _n(e){if(!e)return;let t=Number(e);return Number.isFinite(t)&&t>0?t:void 0}function vn(e,t){let n=e.endsWith(`/`)?e:`${e}/`;return n.endsWith(`/v1/${t}/`)||n.endsWith(`/v1/${t}`)?e:new URL(`v1/${t}`,n).toString()}function yn(e){return fn.has(e)}function bn(e=process.cwd()){let t=x.resolve(e);for(;;){for(let e of dn)if(k(x.join(t,e)))return t;let e=x.dirname(t);if(e===t)return process.cwd();t=e}}function xn(e,t=process.cwd()){try{let n=re(d.resolveRegistryPath(e.PORT_REGISTRY_PATH??u),`utf8`),r=f.parse(JSON.parse(n)),i=p(bn(t)),a=e.NODE_ENV||`development`,o=e.BROWSE_TOOL_OTEL_LOG_SINK_SERVICE_NAME??`log-sink-mcp-http`,s=r.entries.find(e=>e.repositoryPath===i&&e.serviceName===o&&e.serviceType===`tool`&&(e.environment??`development`)===a);if(!s)return;let c=typeof s.metadata?.healthCheckUrl==`string`&&s.metadata.healthCheckUrl.length>0?s.metadata.healthCheckUrl:void 0;return c?new URL(c).origin:`http://${s.host===`0.0.0.0`?`127.0.0.1`:s.host}:${s.port}`}catch{return}}function Sn(e,t){try{let n=new URL(e),r=new URL(t);return!yn(n.hostname)||yn(r.hostname)||(n.hostname=r.hostname),n.toString()}catch{return e}}function Cn(e,t){let n=t===`traces`?e.OTEL_EXPORTER_OTLP_TRACES_ENDPOINT:e.OTEL_EXPORTER_OTLP_LOGS_ENDPOINT;if(n)return n;let r=e.OTEL_EXPORTER_OTLP_ENDPOINT;if(r)return vn(r,t);let i=xn(e);return i?vn(i,t):void 0}function wn(e,t,n){let r=t===`traces`?e.BROWSE_TOOL_OTEL_BROWSER_TRACES_ENDPOINT:e.BROWSE_TOOL_OTEL_BROWSER_LOGS_ENDPOINT;if(r)return Sn(r,n);let i=e.BROWSE_TOOL_OTEL_BROWSER_ENDPOINT;if(i)return Sn(vn(i,t),n);let a=Cn(e,t);return a?Sn(a,n):void 0}function Tn(e,t){let n=e.BROWSE_TOOL_OTEL_BROWSER_SERVICE_NAME??`${e.OTEL_SERVICE_NAME??un}-extension`,r=wn(e,`traces`,t),i=wn(e,`logs`,t);return{enabled:!!(r||i),tracesEndpoint:r,logsEndpoint:i,serviceName:n}}function En(e,t){let n=t===`traces`?e.OTEL_EXPORTER_OTLP_TRACES_HEADERS:e.OTEL_EXPORTER_OTLP_LOGS_HEADERS,r=t===`traces`?e.OTEL_EXPORTER_OTLP_TRACES_TIMEOUT:e.OTEL_EXPORTER_OTLP_LOGS_TIMEOUT;return{url:Cn(e,t),headers:gn(n??e.OTEL_EXPORTER_OTLP_HEADERS),timeoutMillis:_n(r??e.OTEL_EXPORTER_OTLP_TIMEOUT)}}function Dn(e){return e instanceof Error?e.message:String(e)}function On(e){if(e instanceof Error)return hn({"exception.type":e.name,"exception.message":e.message,"exception.stacktrace":e.stack})}let G=class{contextStore=new Ee;tracesEnabled;logsEnabled;tracerProvider;tracer;logSinkReady;constructor(e={}){let t=e.env??process.env,n=pn(t.OTEL_SDK_DISABLED),r=pn(t.BROWSE_TOOL_OTEL_ENABLED),i=En(t,`traces`);this.tracesEnabled=!n&&(r||!!i.url),this.logsEnabled=!n;let a=e.serviceName??t.BROWSE_TOOL_OTEL_SERVICE_NAME??t.OTEL_SERVICE_NAME??un,o=e.serviceVersion??t.npm_package_version,s={"service.name":a};o&&(s[`service.version`]=o);let c=Ne(s);if(this.tracesEnabled){let t=e.traceExporter??new Me(i);this.tracerProvider=new Fe({resource:c,spanProcessors:[e.traceExporter?new Ie(t):new Pe(t)]})}this.tracer=this.tracerProvider?.getTracer(a,o)??je.getTracer(a,o),this.logSinkReady=this.logsEnabled?De({env:t,serviceName:a,serviceVersion:o,workspaceRoot:bn(),enableTraces:!1,enableLogs:!0}).catch(()=>null):Promise.resolve(null)}isEnabled(){return this.tracesEnabled||this.logsEnabled}getActiveTraceContext(){let e=this.contextStore.getStore();if(!e)return{};let t=je.getSpan(e);if(!t)return{};let n=t.spanContext();return{traceId:n.traceId,spanId:n.spanId}}async runInSpan(e,t,n){if(!this.tracesEnabled)return await n(void 0);let r=this.contextStore.getStore()??Ae.active(),i=this.tracer.startSpan(e,{kind:t.kind??Oe.INTERNAL,attributes:hn(t.attributes)},r),a=je.setSpan(r,i);return await this.contextStore.run(a,async()=>{try{return await n(i)}catch(e){throw i.recordException(e instanceof Error?e:Error(Dn(e))),i.setStatus({code:ke.ERROR,message:Dn(e)}),e}finally{i.end()}})}log(e,t,n={}){this.logsEnabled&&this.logSinkReady.then(r=>{if(!r)return;let i=n.context??this.contextStore.getStore()??Ae.active(),a=On(n.exception);r.logger[e](t,{attributes:hn({...n.attributes,...a}),context:i,exception:n.exception})})}async forceFlush(){let e=await this.logSinkReady;await Promise.allSettled([this.tracerProvider?.forceFlush(),e?.flush()])}async shutdown(){let e=await this.logSinkReady;await Promise.allSettled([this.tracerProvider?.shutdown(),e?.shutdown()])}};G=V([v(),z(`design:paramtypes`,[Object])],G);let kn=class{pendingTasks=new Map;taskQueue=[];lastPollAt;lastResultAt;maxQueueSize=100;defaultTimeoutMs=H;constructor(e,t=new G){this.webSocketHub=e,this.telemetry=t}async queueTask(e,t,n=this.defaultTimeoutMs,r){if(this.pendingTasks.size>=this.maxQueueSize)throw Error(`Task queue full (max ${this.maxQueueSize} tasks)`);let i=r??t.browserId,a={id:crypto.randomUUID(),tool:e,arguments:t,createdAt:new Date,timeoutMs:n,browserId:i,pageId:typeof t.pageId==`string`?t.pageId:void 0,telemetry:this.getTaskTelemetryContext()};return new Promise((e,t)=>{let r=setTimeout(()=>{this.pendingTasks.delete(a.id);let e=this.taskQueue.findIndex(e=>e.id===a.id);e!==-1&&this.taskQueue.splice(e,1),t(Error(`Task ${a.id} timed out after ${n}ms`))},n);if(this.pendingTasks.set(a.id,{task:a,resolve:e,reject:t,timeoutId:r}),this.webSocketHub){let e=i?this.pushTaskViaWebSocket(a,i):!1;if(e||=this.pushTaskToAnyConnection(a),e)return}this.taskQueue.push(a)})}pushTaskToAnyConnection(e){if(!this.webSocketHub)return!1;let t=this.webSocketHub.getStats();if(t.totalConnections===0)return!1;let n=t.connections[0];return n?this.pushTaskViaWebSocket(e,n.browserId):!1}pushTaskViaWebSocket(e,t){if(!this.webSocketHub)return!1;let n={type:`task:push`,id:crypto.randomUUID(),payload:{taskId:e.id,tool:e.tool,arguments:e.arguments,telemetry:e.telemetry}};return this.webSocketHub.pushTask(t,n)}getNextTask(){return this.lastPollAt=new Date,this.taskQueue.shift()}submitResult(e){this.lastResultAt=new Date;let t=this.pendingTasks.get(e.taskId);if(t)return clearTimeout(t.timeoutId),this.pendingTasks.delete(e.taskId),t.resolve(e),t.task}getConnectionStatus(){let e=new Date,t=this.lastPollAt?e.getTime()-this.lastPollAt.getTime()<5e3:!1,n=this.webSocketHub?this.webSocketHub.getStats().totalConnections>0:!1;return{connected:t||n,lastPollAt:this.lastPollAt,lastResultAt:this.lastResultAt,pendingTasks:this.pendingTasks.size}}clearAllTasks(e){for(let[t,n]of this.pendingTasks)clearTimeout(n.timeoutId),n.reject(Error(e)),this.pendingTasks.delete(t);this.taskQueue.length=0}get queueSize(){return this.taskQueue.length}get pendingCount(){return this.pendingTasks.size}getTaskTelemetryContext(){let e=this.telemetry.getActiveTraceContext();if(!(!e.traceId&&!e.spanId))return{traceId:e.traceId,parentSpanId:e.spanId}}};kn=V([v(),B(0,_(R.WebSocketHub)),B(0,y()),B(1,_(R.TelemetryService)),B(1,y()),z(`design:paramtypes`,[Object,Object])],kn);const An=new Set(`browser_navigate.browser_go_back.browser_go_forward.browser_reload.browser_click.browser_fill.browser_type.browser_upload_file.browser_select.browser_hover.browser_drag.browser_press_key.browser_inspect_element.browser_resolve_locator.browser_pdf.browser_screenshot.browser_snapshot.browser_list_pages.browser_new_page.browser_select_page.browser_close_page.browser_wait_for.browser_evaluate_script.browser_resize_page.browser_handle_dialog.browser_emulate.browser_list_network_requests.browser_get_network_request.browser_list_console_messages.browser_expect`.split(`.`)),jn=new Set([`browser_launch`,`browser_list_profiles`,`browser_create_profile`,`browser_delete_profile`,`browser_save_profile_state`,`browser_start_trace`,`browser_stop_trace`,`run_spec`,`discover_specs`]);let Mn=class{constructor(e){this.taskQueue=e}isToolSupported(e){return An.has(e)}isToolNotApplicable(e){return jn.has(e)}async executeTool(e,t){if(this.isToolNotApplicable(e))return{content:[{type:`text`,text:`Tool "${e}" is not available in Chrome extension mode. This tool requires Playwright and cannot be run through the extension.`}],isError:!0};if(!this.isToolSupported(e))return{content:[{type:`text`,text:`Tool "${e}" is not yet implemented in Chrome extension mode.`}],isError:!0};if(!this.taskQueue.getConnectionStatus().connected)return{content:[{type:`text`,text:`Chrome extension is not connected. Please ensure the extension is running and connected to the server.`}],isError:!0};try{let n=await this.taskQueue.queueTask(e,t);return n.success?n.result?n.result:{content:[{type:`text`,text:`Tool executed successfully`}]}:{content:[{type:`text`,text:n.error||n.result?.content?.[0]?.text||`Unknown error from Chrome extension`}],isError:!0}}catch(e){return{content:[{type:`text`,text:e instanceof Error?e.message:String(e)}],isError:!0}}}getSupportedTools(){return Array.from(An)}getNotApplicableTools(){return Array.from(jn)}};Mn=V([v(),B(0,_(R.ExtensionTaskQueue)),z(`design:paramtypes`,[Object])],Mn);const Nn=l.min,Pn=`PLAYWRIGHT_PORT`;function K(){return process.env.PLAYWRIGHT_HOST?.trim()||`localhost`}function Fn(){let e=process.env[Pn];if(!e)return Nn;let t=Number.parseInt(e,10);if(Number.isNaN(t)||t<=0||t>65535)throw Error(`Invalid ${Pn} value: ${e}`);return t}function q(e=void 0,t){let n=typeof e==`string`?e:e?.host??K(),r=e&&typeof e!=`string`?e.port:t??Fn();return`http://${n===`0.0.0.0`?`127.0.0.1`:n===`::`?`::1`:n}:${r}`}let In=class{baseUrl;defaultTimeout=H;constructor(){this.baseUrl=q()}setBaseUrl(e){this.baseUrl=e}getBaseUrl(){return this.baseUrl}async execute(e,t={}){let n=`${this.baseUrl}/execute`,r=new AbortController,i=setTimeout(()=>r.abort(),this.defaultTimeout);try{let a=await fetch(n,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({tool:e,arguments:t}),signal:r.signal});if(clearTimeout(i),!a.ok){let e=await a.text();return{content:[{type:`text`,text:`HTTP error ${a.status}: ${e}`}],isError:!0}}let o=await a.json();return o.success?o.result||{content:[{type:`text`,text:`No result returned`}]}:{content:[{type:`text`,text:o.error||o.result?.content?.[0]?.text||`Unknown error`}],isError:!0}}catch(e){return clearTimeout(i),e instanceof Error&&e.name===`AbortError`?{content:[{type:`text`,text:`Request timed out after ${this.defaultTimeout}ms`}],isError:!0}:{content:[{type:`text`,text:e instanceof Error?e.message:String(e)}],isError:!0}}}async isHealthy(){try{let e=await fetch(`${this.baseUrl}/health`,{headers:{"Content-Type":`application/json`}});return e.ok?(await e.json()).status===`healthy`:!1}catch{return!1}}};In=V([v(),z(`design:paramtypes`,[])],In);let Ln=class{defaultTimeout=5e3;async check(e,t=this.defaultTimeout){try{let n=`${q(K(),e)}/health`,r=new AbortController,i=setTimeout(()=>r.abort(),t);try{let t=await fetch(n,{signal:r.signal,headers:{"Content-Type":`application/json`}});if(clearTimeout(i),!t.ok)return{healthy:!1,error:`HTTP error ${t.status}: ${t.statusText}`};let a=await t.json();return a.status===`healthy`?{healthy:!0,port:e,serviceName:a.service,browserCount:a.browsers?.count}:{healthy:!1,error:`Unexpected health status: ${a.status||`unknown`}`}}catch(e){return clearTimeout(i),e instanceof Error?e.name===`AbortError`?{healthy:!1,error:`Health check timed out after ${t}ms`}:`code`in e&&e.code===`ECONNREFUSED`?{healthy:!1,error:`Connection refused - server not running`}:{healthy:!1,error:`Network error: ${e.message}`}:{healthy:!1,error:`Unknown error: ${String(e)}`}}}catch(e){return{healthy:!1,error:e instanceof Error?e.message:String(e)}}}};Ln=V([v()],Ln);var Rn;const zn=ye(import.meta.url),Bn=b.dirname(zn),Vn=[`pnpm-workspace.yaml`,`nx.json`,`.git`],J=`browse-tool-http`,Hn=`tool`,Un=`http-serve`,Wn=`ESRCH`,Gn=`node`,Kn=`cli.mjs`,qn=`cli.ts`,Y=`[HttpServerManager]`;var Jn=class extends Error{code=`HTTP_SERVER_PORT_ERROR`;recovery=`Check if other services are occupying ports in the configured range.`;constructor(e,t){super(e,t),this.name=`HttpServerPortError`}},Yn=class extends Error{code=`HTTP_SERVER_STARTUP_ERROR`;recovery=`Check server logs, ensure CLI binary exists, and verify port availability.`;constructor(e,t){super(e,t),this.name=`HttpServerStartupError`}},Xn=class extends Error{code=`HTTP_SERVER_STOP_ERROR`;recovery=`Manually kill the process and clean up the port registry.`;causes;constructor(e,t=[],n){super(e,n),this.name=`HttpServerStopError`,this.causes=t}};async function Zn(e){try{return await ae.access(e),!0}catch(e){if(e instanceof Error&&`code`in e&&e.code===`ENOENT`)return!1;throw e}}const Qn=Le(_e);async function $n(e){try{let{stdout:t}=await Qn(`git`,[`rev-parse`,`--git-common-dir`],{cwd:e}),n=t.trim(),r=b.isAbsolute(n)?n:b.join(e,n),i=b.dirname(r);return i===e?null:i}catch{return null}}async function er(e=process.cwd()){let t=b.resolve(e);for(;;){for(let e of Vn)if(await Zn(b.join(t,e)))return t;let e=b.dirname(t);if(e===t)return process.cwd();t=e}}let tr=class{repositoryPath;environment;host;constructor(e,t,n=new m(process.env.PROCESS_REGISTRY_PATH)){this.healthCheck=e,this.portRegistry=t,this.processRegistry=n,this.environment=process.env.NODE_ENV||`development`,this.host=K()}async getRepositoryPath(){return this.repositoryPath||=await er(process.cwd()),this.repositoryPath}buildErrorStatus(e){return{running:!1,error:e instanceof Error?e.message:String(e),errorCode:e instanceof Error&&`code`in e?e.code:void 0,errorRecovery:e instanceof Error&&`recovery`in e?e.recovery:void 0,errorCause:e instanceof Error&&e.cause instanceof Error?e.cause.message:void 0}}async listServiceEntries(){let e=await this.getRepositoryPath(),t={serviceName:J,serviceType:Hn,environment:this.environment},n=await this.portRegistry.listAllocations({...t,repositoryPath:e});if(n.length>0)return n;let r=await $n(e);if(r){let e=await this.portRegistry.listAllocations({...t,repositoryPath:r});if(e.length>0)return e}return this.portRegistry.listAllocations(t)}async releaseEntry(e){if(e.pid){let t=await this.processRegistry.releaseProcess({repositoryPath:e.repositoryPath,serviceName:J,serviceType:`service`,environment:e.environment,pid:e.pid,kill:!1,releasePort:!1});!t.success&&!t.error?.includes(`No matching process entry`)&&console.error(`${Y} Failed to release process entry for port ${e.port}: ${t.error}`)}let t=await this.portRegistry.releasePort({repositoryPath:e.repositoryPath,serviceName:J,serviceType:Hn,environment:e.environment,pid:e.pid});!t.success&&!t.error?.includes(`No matching registry entry`)&&console.error(`${Y} Failed to release entry for port ${e.port}: ${t.error}`)}async discoverHealthyService(){let e=await this.listServiceEntries();for(let t of e)try{let e=await this.healthCheck.check(t.port);if(e.healthy&&e.serviceName===J)return{entry:t,browserCount:e.browserCount}}catch(e){console.error(`${Y} Health check threw for port ${t.port}: ${e instanceof Error?e.message:String(e)}`)}return null}async discoverHealthyServiceOnPort(e){let t=await this.findRegistrationByPort(e);if(!t)return null;try{let n=await this.healthCheck.check(e);if(n.healthy&&n.serviceName===J)return{entry:t,browserCount:n.browserCount}}catch(t){console.error(`${Y} Health check threw for requested port ${e}: ${t instanceof Error?t.message:String(t)}`)}return null}async cleanupStaleEntries(){let e=await this.listServiceEntries(),t=[];for(let n of e)try{let e;try{e=await this.healthCheck.check(n.port)}catch(e){console.error(`${Y} Health check threw for port ${n.port}, skipping cleanup: ${e instanceof Error?e.message:String(e)}`);continue}e.healthy&&e.serviceName===J||(n.pid&&await this.killProcess(n.pid),await this.releaseEntry(n))}catch(e){let r=`Failed to clean up entry on port ${n.port}: ${e instanceof Error?e.message:String(e)}`;console.error(`${Y} ${r}`),t.push(r)}t.length>0&&console.error(`${Y} ${t.length} entry cleanup failure(s) during stale entry removal`)}async findRegistrationByPort(e){return(await this.listServiceEntries()).find(t=>t.port===e)??null}async findAvailablePort(e,t={}){let n=await this.getRepositoryPath(),r=await this.portRegistry.findAvailablePort({repositoryPath:n,serviceName:J,serviceType:Hn,environment:this.environment,host:this.host,preferredPort:e,portRange:t.exactPort?{min:e,max:e}:void 0});if(!r.success||!r.port)throw new Jn(`No available port found from ${e}: ${r.error}`);return r.port}async ensureExactPortAvailable(e){let t=await this.findAvailablePort(e,{exactPort:!0});if(t!==e)throw new Jn(`Requested port ${e} is unavailable; next available port is ${t}`)}async resolveCliCommand(e){let t=b.resolve(Bn,Kn),n=b.resolve(Bn,`..`,qn),r=[`--port`,e.toString(),`--host`,this.host];if(await Zn(t))return[Gn,[t,Un,...r]];if(await Zn(n))return[`bun`,[`run`,n,Un,...r]];let i=b.resolve(Bn,`..`,`..`),a=b.join(i,`dist`,Kn),o=b.join(i,`src`,qn);if(await Zn(a))return[Gn,[a,Un,...r]];if(await Zn(o))return[`bun`,[`run`,o,Un,...r]];throw new Yn(`Cannot find CLI at ${t}, ${n}, ${a}, or ${o}`)}async startHttpServer(e){let[t,n]=await this.resolveCliCommand(e),r=F(t,n,{detached:!0,stdio:`ignore`,env:{...process.env,NODE_ENV:this.environment,PLAYWRIGHT_SKIP_DYNAMIC_TOOL_COMMANDS:`1`}}),i=await new Promise(e=>{let t=setTimeout(()=>{r.removeAllListeners(`error`),e(null)},100);r.once(`error`,n=>{clearTimeout(t),e(n)})});if(i)throw new Yn(`Failed to spawn HTTP server process: ${i.message}`,{cause:i});r.unref();let a=r.pid;if(!a)throw new Yn(`Failed to spawn HTTP server - no PID returned`);return a}async killProcess(e){try{process.kill(e,0),process.kill(e,`SIGTERM`)}catch(t){if((t instanceof Error&&`code`in t?t.code:void 0)===Wn){console.error(`${Y} Process ${e} does not exist — skipping kill`);return}throw t}await new Promise(e=>setTimeout(e,1e3));try{process.kill(e,0),process.kill(e,`SIGKILL`)}catch(t){if((t instanceof Error&&`code`in t?t.code:void 0)===Wn){console.error(`${Y} Process ${e} exited after SIGTERM`);return}throw t}}async cleanupFailedSpawn(e,t){await this.killProcess(e);let n=await this.findRegistrationByPort(t);n&&n.pid===e&&await this.releaseEntry(n)}async attemptSpawn(e){let t=await this.findAvailablePort(e),n=await this.startHttpServer(t);await new Promise(e=>setTimeout(e,1e4));let r;try{r=await this.healthCheck.check(t)}catch(e){let r=e instanceof Error?e.message:String(e);return console.error(`${Y} Health check threw on port ${t}: ${r}`),await this.cleanupFailedSpawn(n,t),{success:!1,reason:`health_check_error`,port:t,error:r}}if(!r.healthy)return await this.cleanupFailedSpawn(n,t),{success:!1,reason:`health_check_failed`,port:t,error:r.error};if(r.serviceName!==J)return await this.cleanupFailedSpawn(n,t),{success:!1,reason:`service_name_mismatch`,port:t,error:`Expected ${J}, got ${r.serviceName}`};let i=await this.findRegistrationByPort(t);return!i||i.pid!==n?(console.error(`${Y} Ownership mismatch on port ${t}: expected pid ${n}, found ${i?.pid}`),await this.cleanupFailedSpawn(n,t),{success:!1,reason:`ownership_mismatch`,port:t}):(console.error(`${Y} Server self-registered on port ${t} (pid: ${i.pid})`),{success:!0,port:t,pid:n,browserCount:r.browserCount})}async ensureRunning(e=Fn(),t={}){try{if(t.exactPort){let t=await this.discoverHealthyServiceOnPort(e);if(t)return{running:!0,port:t.entry.port,pid:t.entry.pid,browserCount:t.browserCount,spawned:!1};await this.cleanupStaleEntries(),await this.ensureExactPortAvailable(e);let n=await this.attemptSpawn(e);if(n.success)return{running:!0,port:n.port,pid:n.pid,browserCount:n.browserCount,spawned:!0};let r=new Yn(`Failed to start browse-tool HTTP server on requested port ${e}: ${n.reason}`);return this.buildErrorStatus(r)}let n=await this.discoverHealthyService();if(n)return{running:!0,port:n.entry.port,pid:n.entry.pid,browserCount:n.browserCount,spawned:!1};await this.cleanupStaleEntries();let r=e;for(let e=0;e<3;e+=1){let t=await this.attemptSpawn(r);if(t.success)return{running:!0,port:t.port,pid:t.pid,browserCount:t.browserCount,spawned:!0};console.error(`${Y} Spawn attempt ${e+1} failed: ${t.reason} on port ${t.port}`),r=t.port+1}let i=new Yn(`Failed to start browse-tool HTTP server after 3 attempts`);return this.buildErrorStatus(i)}catch(e){return this.buildErrorStatus(e)}}async stop(){let e=await this.listServiceEntries();if(e.length===0)return!1;let t=[];for(let n of e)try{n.pid&&await this.killProcess(n.pid),await this.releaseEntry(n)}catch(e){let r=e instanceof Error?e:Error(`Failed to stop entry on port ${n.port}: ${String(e)}`);console.error(`${Y} ${r.message}`),t.push(r)}if(t.length>0){let n=t.map(e=>e.message);throw new Xn(`Failed to stop ${t.length} of ${e.length} server(s): ${n.join(`; `)}`,t,{cause:t[0]})}return!0}async getStatus(){try{let e=await this.discoverHealthyService();if(e)return{running:!0,port:e.entry.port,pid:e.entry.pid,browserCount:e.browserCount};let t=await this.listServiceEntries();if(t.length>0){let e=t[0];return{running:!1,port:e.port,pid:e.pid,error:`Server registered but not healthy`,errorCode:`HTTP_SERVER_STALE_ENTRY`,errorRecovery:`Run ensureRunning() to start a fresh server or clean up stale entries.`}}return{running:!1,error:`No HTTP server registered`,errorCode:`HTTP_SERVER_NOT_REGISTERED`,errorRecovery:`Call ensureRunning() to discover or spawn a server.`}}catch(e){return this.buildErrorStatus(e)}}};tr=V([v(),B(0,_(R.HttpServerHealthCheck)),B(1,_(R.PortRegistryService)),B(2,_(R.ProcessRegistryService)),z(`design:paramtypes`,[Object,Object,typeof(Rn=m!==void 0&&m)==`function`?Rn:Object])],tr);const nr=900*1e3,rr=1800*1e3,ir=`PLAYWRIGHT_BROWSER_IDLE_TIMEOUT_MINUTES`;function ar(){let e=process.env[ir];if(!e)return rr;let t=Number.parseFloat(e);return!Number.isFinite(t)||t<=0?rr:Math.round(t*60*1e3)}let or=class{intervalId=null;browserIdleTimeoutMs=ar();constructor(e,t){this.browserService=e,this.pageRegistry=t}start(){this.intervalId||(this.intervalId=setInterval(()=>{this.cleanup().catch(e=>{console.warn(`[IdleCleanup] Error during cleanup:`,e)})},6e4),this.intervalId&&typeof this.intervalId==`object`&&`unref`in this.intervalId&&this.intervalId.unref(),console.warn(`[IdleCleanup] Started (page timeout: ${nr/6e4}min, browser timeout: ${this.browserIdleTimeoutMs/6e4}min)`))}stop(){this.intervalId&&(clearInterval(this.intervalId),this.intervalId=null,console.warn(`[IdleCleanup] Stopped`))}async cleanup(){let e=Date.now();await this.cleanupIdlePages(e),await this.cleanupIdleBrowsers(e)}async cleanupIdlePages(e){let t=this.pageRegistry.list();for(let n of t){let t=e-n.lastAccessedAt.getTime();if(!(t<nr)){console.warn(`[IdleCleanup] Closing idle page "${n.id}" (idle ${Math.round(t/6e4)}min)`);try{if(n.page)await n.page.close();else{let e=this.browserService.getBrowser(n.browserId);if(e&&(e.pageIds.delete(n.id),e.currentPageId===n.id)){let t=Array.from(e.pageIds);e.currentPageId=t.length>0?t[0]:null}this.pageRegistry.remove(n.id)}}catch(e){console.warn(`[IdleCleanup] Failed to close page "${n.id}":`,e)}}}}async cleanupIdleBrowsers(e){let t=this.browserService.listBrowsers();for(let n of t){let t=n.specOrigin?nr:this.browserIdleTimeoutMs,r=e-n.lastAccessedAt.getTime();if(!(r<t)){console.warn(`[IdleCleanup] Closing idle browser "${n.id}" (idle ${Math.round(r/6e4)}min)`);try{await this.browserService.closeBrowser(n.id)}catch(e){console.warn(`[IdleCleanup] Failed to close browser "${n.id}":`,e)}}}}};or=V([v(),B(0,_(R.BrowserService)),B(1,_(R.PageRegistry)),z(`design:paramtypes`,[Object,Object])],or);let sr=class{constructor(e){this.portRegistry=e}async findAvailablePort(e){let t=await this.portRegistry.findAvailablePort({repositoryPath:e.repositoryPath,serviceName:`browse-tool-http`,serviceType:`tool`,environment:e.environment,host:e.host,preferredPort:e.preferredPort,portRange:e.exactPort?{min:e.preferredPort,max:e.preferredPort}:void 0});if(!t.success||!t.port)throw Error(t.error||`Failed to acquire HTTP port from ${e.preferredPort}`);return t.port}async acquirePorts(e){let t,n=!1;try{if(e.reserveMcpPort){let n=e.preferredMcpPort??l.min,r=await this.portRegistry.reservePort({repositoryPath:e.repositoryPath,serviceName:e.mcpServiceName??`browse-tool-mcp-http`,serviceType:e.mcpServiceType??`tool`,environment:e.environment,preferredPort:n,pid:e.pid,host:e.host,force:!0,portRange:{min:n,max:n},metadata:e.mcpMetadata});if(!r.success||!r.record)throw Error(r.error||`Failed to reserve MCP port ${n}`);t=r.record}return{httpPort:await this.findAvailablePort({repositoryPath:e.repositoryPath,environment:e.environment,host:e.host,preferredPort:e.preferredHttpPort??l.min,exactPort:e.exactHttpPort}),mcpPort:t?.port,release:async()=>{n||!t||(n=!0,await this.portRegistry.releasePort({repositoryPath:t.repositoryPath,serviceName:t.serviceName,serviceType:t.serviceType,environment:t.environment,pid:t.pid}))}}}catch(e){throw t&&(n=!0,await this.portRegistry.releasePort({repositoryPath:t.repositoryPath,serviceName:t.serviceName,serviceType:t.serviceType,environment:t.environment,pid:t.pid})),e}}};sr=V([v(),B(0,_(R.PortRegistryService)),z(`design:paramtypes`,[Object])],sr);let cr=class{browsers=new Map;pages=new Map;trackBrowser(e,t){this.browsers.has(e)||this.browsers.set(e,{browserId:e,mode:t,createdAt:new Date,pageIds:new Set})}trackPage(e,t,n){if(this.pages.has(e))return;this.pages.set(e,{pageId:e,browserId:t,url:n,createdAt:new Date});let r=this.browsers.get(t);r&&r.pageIds.add(e)}untrackBrowser(e){let t=this.browsers.get(e);if(t){for(let e of t.pageIds)this.pages.delete(e);this.browsers.delete(e)}}untrackPage(e){let t=this.pages.get(e);if(t){let n=this.browsers.get(t.browserId);n&&n.pageIds.delete(e),this.pages.delete(e)}}getBrowserIds(){return Array.from(this.browsers.keys())}getPageIds(){return Array.from(this.pages.keys())}getPageIdsForBrowser(e){let t=this.browsers.get(e);return t?Array.from(t.pageIds):[]}getSessionState(){return{browsers:Array.from(this.browsers.values()).map(e=>({browserId:e.browserId,mode:e.mode,createdAt:e.createdAt.toISOString(),pageIds:Array.from(e.pageIds)})),pages:Array.from(this.pages.values()).map(e=>({pageId:e.pageId,browserId:e.browserId,url:e.url,createdAt:e.createdAt.toISOString()})),totalBrowsers:this.browsers.size,totalPages:this.pages.size}}clear(){this.browsers.clear(),this.pages.clear()}};cr=V([v()],cr);let lr=class{pageStates=new Map;pageInstances=new Map;startMonitoring(e,t){this.stopMonitoring(e);let n={networkRequests:new Map,consoleMessages:[],requestCounter:0,messageCounter:0,listeners:{}};n.listeners.request=e=>{let t=`req-${++n.requestCounter}`,r={id:t,url:e.url(),method:e.method(),resourceType:e.resourceType(),headers:e.headers(),postData:e.postData()??void 0,timestamp:new Date};n.networkRequests.set(t,r)},n.listeners.response=e=>{let t=e.request();for(let[,r]of n.networkRequests)if(r.url===t.url()&&!r.response){r.response={status:e.status(),statusText:e.statusText(),headers:e.headers(),timing:e.request().timing().responseEnd};break}},n.listeners.console=e=>{let t=`msg-${++n.messageCounter}`,r=e.location(),i={id:t,type:e.type(),text:e.text(),location:r.url?{url:r.url,lineNumber:r.lineNumber,columnNumber:r.columnNumber}:void 0,timestamp:new Date};n.consoleMessages.push(i)},t.on(`request`,n.listeners.request),t.on(`response`,n.listeners.response),t.on(`console`,n.listeners.console),this.pageStates.set(e,n),this.pageInstances.set(e,t),t.once(`close`,()=>{this.stopMonitoring(e)})}stopMonitoring(e){let t=this.pageStates.get(e),n=this.pageInstances.get(e);t&&n&&(t.listeners.request&&n.off(`request`,t.listeners.request),t.listeners.response&&n.off(`response`,t.listeners.response),t.listeners.console&&n.off(`console`,t.listeners.console)),this.pageStates.delete(e),this.pageInstances.delete(e)}getNetworkRequests(e){let t=this.pageStates.get(e);return t?Array.from(t.networkRequests.values()):[]}getNetworkRequest(e,t){let n=this.pageStates.get(e);if(n)return n.networkRequests.get(t)}getConsoleMessages(e,t){let n=this.pageStates.get(e);return n?t?n.consoleMessages.filter(e=>e.type===t):[...n.consoleMessages]:[]}clearData(e){let t=this.pageStates.get(e);t&&(t.networkRequests.clear(),t.consoleMessages=[],t.requestCounter=0,t.messageCounter=0)}};lr=V([v()],lr);let ur=class{pages=new Map;pageIdCounter=0;constructor(e){this.webSocketHub=e}async register(e){let{page:t,browser:n,context:r,browserId:i,profileName:a,mode:o,onClose:s}=e,c=`page-${++this.pageIdCounter}`,l={id:c,mode:o??`playwright`,page:t,browser:n,context:r,browserId:i,url:t.url(),title:await t.title(),profileName:a,createdAt:new Date,lastAccessedAt:new Date,onClose:s};return this.pages.set(c,l),t.once(`close`,()=>{try{s?.(c)}catch{}this.remove(c)}),c}registerExtensionPage(e,t,n,r=!0){let i=`page-${++this.pageIdCounter}`,a={id:i,mode:`extension`,page:void 0,browser:void 0,context:void 0,browserId:e,url:n??`about:blank`,title:`Extension Tab`,extensionTabId:t,createdAt:new Date,lastAccessedAt:new Date};return this.pages.set(i,a),r&&this.webSocketHub?.broadcastPageCreated(e,i,n),i}get(e){return this.pages.get(e)}remove(e){let t=this.pages.get(e);t&&(this.pages.delete(e),t.mode===`extension`&&this.webSocketHub?.broadcastPageRemoved(t.browserId,e))}list(){return Array.from(this.pages.values())}findByBrowser(e){return this.list().filter(t=>t.browserId===e)}findByProfile(e){return this.list().filter(t=>t.profileName===e)}touchPage(e){let t=this.pages.get(e);t&&(t.lastAccessedAt=new Date)}async updateMetadata(e){let t=this.get(e);t&&t.page&&(t.url=t.page.url(),t.title=await t.page.title())}clear(){this.pages.clear(),this.pageIdCounter=0}};ur=V([v(),B(0,_(R.WebSocketHub)),B(0,y()),z(`design:paramtypes`,[Object])],ur);let dr=class{pauseStates=new Map;getOrCreateState(e){let t=this.pauseStates.get(e);return t||(t={isPaused:!1,stepName:null,reason:null,pausedAt:null,resolver:null},this.pauseStates.set(e,t)),t}pause(e,t,n){let r=this.getOrCreateState(e);if(r.isPaused)throw Error(`Session "${e}" is already paused at step "${r.stepName}"`);return r.isPaused=!0,r.stepName=t,r.reason=n??null,r.pausedAt=new Date,new Promise(e=>{r.resolver=e})}resume(e){let t=this.pauseStates.get(e);if(!t||!t.isPaused)return!1;let n=t.resolver;return t.isPaused=!1,t.stepName=null,t.reason=null,t.pausedAt=null,t.resolver=null,n&&n(),!0}waitForResume(e){let t=this.pauseStates.get(e);return!t||!t.isPaused||!t.resolver?Promise.resolve():new Promise(e=>{let n=t.resolver;t.resolver=()=>{n&&n(),e()}})}getStatus(e){let t=this.pauseStates.get(e);return{sessionId:e,isPaused:t?.isPaused??!1,stepName:t?.stepName??null,reason:t?.reason??null,pausedAt:t?.pausedAt??null}}isPaused(e){return this.pauseStates.get(e)?.isPaused??!1}clear(e){let t=this.pauseStates.get(e);t?.resolver&&t.resolver(),this.pauseStates.delete(e)}};dr=V([v()],dr);let fr=class{profilesDir;constructor(){this.profilesDir=process.env.PLAYWRIGHT_PROFILES_DIR||T(he(),`.browse-tool`,`profiles`)}getProfilesDir(){return this.profilesDir}getProfileDir(e){return T(this.getProfilesDir(),e)}getProfilePath(e){return T(this.getProfileDir(e),`profile.json`)}getStorageStatePath(e){return T(this.getProfileDir(e),`storage-state.json`)}async ensureProfilesDir(){let e=this.getProfilesDir();k(e)||await A(e,{recursive:!0})}async list(){await this.ensureProfilesDir();let e=await ce(this.getProfilesDir(),{withFileTypes:!0}),t=[];for(let n of e)if(n.isDirectory()){let e=await this.get(n.name);e&&t.push(e)}return t}async get(e){let t=this.getProfilePath(e);if(!k(t))return null;let n=await j(t,`utf-8`);return JSON.parse(n)}async create(e){if(await this.get(e.name))throw Error(`Profile "${e.name}" already exists`);await A(this.getProfileDir(e.name),{recursive:!0});let t=new Date().toISOString(),n={...e,createdAt:t,updatedAt:t};return await M(this.getProfilePath(e.name),JSON.stringify(n,null,2)),n}async update(e,t){let n=await this.get(e);if(!n)throw Error(`Profile "${e}" not found`);let r={...n,...t,name:n.name,createdAt:n.createdAt,updatedAt:new Date().toISOString()};return await M(this.getProfilePath(e),JSON.stringify(r,null,2)),r}async delete(e){if(!await this.get(e))throw Error(`Profile "${e}" not found`);await ue(this.getProfileDir(e),{recursive:!0})}async saveStorageState(e,t){if(!await this.get(e))throw Error(`Profile "${e}" not found`);await M(this.getStorageStatePath(e),JSON.stringify(t,null,2))}async loadStorageState(e){let t=this.getStorageStatePath(e);if(!k(t))return null;let n=await j(t,`utf-8`);return JSON.parse(n)}};fr=V([v(),z(`design:paramtypes`,[])],fr);const pr=`BROWSE_TOOL_PROXY_CONFIG_DIR`,mr=`session`,hr=/\$\{([A-Z0-9_]+)\}/gi;function gr(e){return e.PLAYWRIGHT_PROFILES_DIR||x.join(he(),`.browse-tool`,`profiles`)}function _r(e){let t=e.env??process.env,n=e.proxyConfigDir??t.BROWSE_TOOL_PROXY_CONFIG_DIR;if(n)return x.resolve(n);let r=e.profilesDir??gr(t);return x.join(x.dirname(x.resolve(r)),`proxy-config`)}function vr(e){if(!e||e.includes(`/`)||e.includes(`\\`)||e===`.`||e===`..`)throw Error(`Invalid proxy config name: ${e}`)}function yr(e,t){vr(t);let n=x.resolve(e),r=x.resolve(n,`${t}.yaml`),i=x.relative(n,r);if(i.startsWith(`..`)||x.isAbsolute(i))throw Error(`Proxy config path escapes configured directory: ${t}`);return r}function br(e,t,n){return e.replace(hr,(e,r)=>{let i=t[r];if(i===void 0)throw Error(`Missing environment variable "${r}" referenced by proxy config ${n}`);return i})}function xr(e,t,n){let r=yr(e,t);if(!k(r))return;let i=Re(re(r,`utf8`));if(!i||typeof i!=`object`||Array.isArray(i))throw Error(`Proxy config ${r} must be a YAML object`);let a=i,o={server:Sr(a,`server`,r,n)},s=Cr(a,`username`,r,n),c=Cr(a,`password`,r,n),l=Cr(a,`bypass`,r,n);return s!==void 0&&(o.username=s),c!==void 0&&(o.password=c),l!==void 0&&(o.bypass=l),o}function Sr(e,t,n,r){let i=Cr(e,t,n,r);if(!i)throw Error(`Proxy config ${n} must define a non-empty "${t}" string`);return i}function Cr(e,t,n,r){let i=e[t];if(i!=null){if(typeof i!=`string`)throw Error(`Proxy config ${n} field "${t}" must be a string`);return br(i,r,n)}}function wr(e){return!!e&&e!==!0&&typeof e==`object`}let X=class{getProxyConfigDir(e={}){return _r(e)}resolve(e){if(wr(e.proxy))return e.proxy;let t=e.env??process.env,n=_r(e);if(e.profileName){let r=xr(n,e.profileName,t);if(r)return r}if(e.proxy===!0){let e=xr(n,mr,t);if(!e)throw Error(`Proxy requested but ${yr(n,mr)} was not found`);return e}return xr(n,`default`,t)}};X=V([v()],X);function Tr(e){return new X().resolve(e)}let Er=class{constructor(e){this.httpClient=e}async execute(e,t={}){return this.httpClient.execute(e,t)}async isServerAvailable(){return this.httpClient.isHealthy()}setServerPort(e){this.httpClient.setBaseUrl(q(void 0,e))}async listTools(){let e=this.httpClient.getBaseUrl(),t=await fetch(`${e}/tools`);if(!t.ok)throw Error(`Failed to list tools: ${t.statusText}`);let n=await t.json();if(n.error)throw Error(n.error);return n.tools}};Er=V([v(),B(0,_(R.HttpBrowserClient)),z(`design:paramtypes`,[Object])],Er);let Dr=class{constructor(e){this.bundler=e}async runSetup(e,t,n){let r=null;try{r=await this.bundler.bundle(e);let i=await import(`${r.outputPath}?t=${Date.now()}`),a=await this.getSetupFunction(i,t,e)(n);return{state:this.validateSetupResult(a,e),success:!0}}catch(t){return{state:{},success:!1,error:`Setup failed for "${e}": ${t instanceof Error?t.message:String(t)}`}}finally{r&&await r.cleanup()}}getSetupFunction(e,t,n){if(t in e&&typeof e[t]==`function`)return e[t];if(t===`default`&&`default`in e){let t=e.default;if(typeof t==`function`)return t;throw Error(`Default export in "${n}" is not a function`)}let r=Object.keys(e).filter(t=>typeof e[t]==`function`);throw r.length===0?Error(`No function exports found in "${n}"`):Error(`Export "${t}" not found in "${n}". Available functions: ${r.join(`, `)}`)}validateSetupResult(e,t){if(e==null)return{};if(typeof e!=`object`||Array.isArray(e))throw Error(`Setup function in "${t}" must return an object or undefined, got ${typeof e}`);return e}};Dr=V([v(),B(0,_(R.SpecBundlerService)),z(`design:paramtypes`,[Object])],Dr);const Z=`[SpecBundler]`,Or=RegExp(`^@playwright\\/test$`),kr=`__PLAYWRIGHT_MCP_STUB_PATH__`,Ar={OUTDIR_NAME:`browse-tool-bundles`,PLUGIN_NAME:`playwright-test-alias`,EXTERNALS:[`playwright`,`playwright/test`,`playwright-core`]},jr={BUILD_FAILED:`SPEC_BUNDLER_BUILD_FAILED`,NO_OUTPUT:`SPEC_BUNDLER_NO_OUTPUT`,IO_FAILED:`SPEC_BUNDLER_IO_FAILED`},Q={UNEXPECTED_BUILD:`Unexpected build exception`,IO_MKDIR:`Failed to create output directory`,CLEANUP_FAILED:`Failed to clean up temporary file`};var Mr=class extends Error{code=jr.BUILD_FAILED;recovery=`Check the spec file for syntax errors and ensure all imports are resolvable.`;constructor(e,t,n){super(`Failed to bundle spec "${S(e)}": ${t}`,n),this.specPath=e,this.buildErrors=t,this.name=`SpecBundlerBuildError`}},Nr=class extends Error{code=jr.IO_FAILED;recovery=`Check filesystem permissions and available disk space.`;constructor(e,t,n){super(`${e} at "${t}"`,n),this.operation=e,this.targetPath=t,this.name=`SpecBundlerIoError`}};let Pr=class{stubPath;nodeModulesPath;constructor(){let e=C(import.meta.url.replace(`file://`,``));this.stubPath=T(e,`stubs`,`playwright-test.mjs`),this.nodeModulesPath=this.findNodeModules(e)}findNodeModules(e){let t=e,n=D(`/`);for(;t!==n;){let e=T(t,`node_modules`);if(k(T(e,`playwright`)))return e;t=C(t)}for(t=e;t!==n;){let e=T(t,`node_modules`);if(k(e))return e;t=C(t)}return T(e,`node_modules`)}async bundle(e){let t=T(P(),Ar.OUTDIR_NAME);try{await A(t,{recursive:!0})}catch(e){throw console.error(`${Z} ${Q.IO_MKDIR} "${t}":`,e),new Nr(Q.IO_MKDIR,t,{cause:e})}let n=T(t,`node_modules`);if(!k(n))try{await fe(this.nodeModulesPath,n,`junction`)}catch{}return this.bundleWithEsbuild(e,t)}async bundleWithEsbuild(e,t){let n=S(e).replace(/\.[^.]+$/,``),r=this.stubPath;try{let i=await ze({entryPoints:[e],outdir:t,bundle:!0,platform:`node`,format:`esm`,target:`node18`,banner:{js:`import { createRequire as __esbuildCreateRequire } from 'module'; const require = __esbuildCreateRequire(import.meta.url);`},external:[...Ar.EXTERNALS],define:{[kr]:JSON.stringify(r)},plugins:[{name:Ar.PLUGIN_NAME,setup(e){e.onResolve({filter:Or},()=>({path:r}))}}],write:!0,logLevel:`silent`});if(i.errors.length>0){let t=i.errors.map(e=>e.text).join(`
|
|
10
|
+
`);throw console.error(`${Z} Build failed for "${e}": ${t}`),new Mr(e,t,{cause:Error(t)})}let a=T(t,`${n}.js`);return{outputPath:a,cleanup:()=>this.safeUnlink(a)}}catch(t){if(t instanceof Mr)throw t;let n=t instanceof Error?t.message:Q.UNEXPECTED_BUILD;throw console.error(`${Z} Build failed for "${e}":`,t),new Mr(e,n,{cause:t})}}async bundleWithHooks(e){let t=T(P(),Ar.OUTDIR_NAME);try{await A(t,{recursive:!0})}catch(e){throw console.error(`${Z} ${Q.IO_MKDIR} "${t}":`,e),new Nr(Q.IO_MKDIR,t,{cause:e})}let n=T(t,`node_modules`);if(!k(n))try{await fe(this.nodeModulesPath,n,`junction`)}catch{}return this.bundleSpecWithHooksEntry(e.specPath,e.hooksPath,t)}async bundleSpecWithHooksEntry(e,t,n){let r=C(D(t)),i=this.stubPath,a=[`process.env.__HOOKS_SOURCE_DIR__ = ${JSON.stringify(r)};`,`import { setup as __hooksSetup } from ${JSON.stringify(t)};`,`await __hooksSetup();`,`delete process.env.__HOOKS_SOURCE_DIR__;`,`export * from ${JSON.stringify(e)};`].join(`
|
|
11
|
+
`);try{let t=await ze({stdin:{contents:a,resolveDir:C(e),loader:`ts`,sourcefile:`spec-with-hooks.ts`},outdir:n,bundle:!0,platform:`node`,format:`esm`,target:`node18`,banner:{js:`import { createRequire as __esbuildCreateRequire } from 'module'; const require = __esbuildCreateRequire(import.meta.url);`},external:[...Ar.EXTERNALS],define:{[kr]:JSON.stringify(i)},plugins:[{name:Ar.PLUGIN_NAME,setup(e){e.onResolve({filter:Or},()=>({path:i}))}}],write:!0,logLevel:`silent`});if(t.errors.length>0){let n=t.errors.map(e=>e.text).join(`
|
|
12
|
+
`);throw console.error(`${Z} Build failed for "${e}" (with hooks): ${n}`),new Mr(e,n,{cause:Error(n)})}let r=T(n,`stdin.js`);return{outputPath:r,cleanup:()=>this.safeUnlink(r)}}catch(t){if(t instanceof Mr)throw t;let n=t instanceof Error?t.message:Q.UNEXPECTED_BUILD;throw console.error(`${Z} Build failed for "${e}" (with hooks):`,t),new Mr(e,n,{cause:t})}}async safeUnlink(e){try{await pe(e)}catch(t){console.warn(`${Z} ${Q.CLEANUP_FAILED} "${e}":`,t)}}};Pr=V([v(),z(`design:paramtypes`,[])],Pr);let Fr=class{async discoverProjects(e){let t=e||process.cwd(),n=[],r=await this.findPlaywrightConfigs(t);for(let e of r){let t=C(e),r={name:S(t),path:t,configPath:e,testDir:await this.getTestDirFromConfig(e),specs:[]};r.specs=await this.getProjectSpecs(t),n.push(r)}return n}async getProjectSpecs(e){let t=T(e,`playwright.config.ts`);if(!k(t))throw Error(`No playwright.config.ts found in ${e}`);let n=T(e,await this.getTestDirFromConfig(t));if(!k(n))return[];let r=[],i=await this.findSpecFiles(n);for(let t of i){let n=await this.getSpecInfo(t,e);r.push(n)}return r}async getSpecInfo(e,t){let n=D(e);if(!k(n))throw Error(`Spec file not found: ${e}`);let r=S(n);return{name:r,path:n,relativePath:t?E(t,n):r,hasArgsSchema:await this.checkHasArgsSchema(n)}}async filterSpecs(e,t){let n=await this.discoverProjects(e),r=[];for(let e of n){if(t?.projectName&&e.name!==t.projectName)continue;let n=e.specs;if(t?.specPattern){let e=this.globToRegex(t.specPattern);n=n.filter(t=>e.test(t.name)||e.test(t.relativePath))}r=r.concat(n)}return r}async findPlaywrightConfigs(e){let t=[];return await this.walkDirectory(e,(e,n)=>{if(n){let t=S(e);return!(t===`node_modules`||t.startsWith(`.`))}return S(e)===`playwright.config.ts`&&t.push(e),!0}),t}async findSpecFiles(e){let t=[];return await this.walkDirectory(e,(e,n)=>n?S(e)!==`node_modules`:(e.endsWith(`.spec.ts`)&&t.push(e),!0)),t}async walkDirectory(e,t){if(!k(e))return;let n=await ce(e,{withFileTypes:!0});for(let r of n){let n=T(e,r.name);r.isDirectory()?t(n,!0)&&await this.walkDirectory(n,t):t(n,!1)}}async getTestDirFromConfig(e){try{let t=(await j(e,`utf-8`)).match(/testDir:\s*['"`]([^'"`]+)['"`]/);if(t)return t[1].replace(/^\.\//,``)}catch{}return`tests`}async checkHasArgsSchema(e){try{let t=await j(e,`utf-8`);return/export\s+(const\s+)?argsSchema\b/.test(t)}catch{return!1}}globToRegex(e){let t=e.replace(/[.+^${}()|[\]\\]/g,`\\$&`).replace(/\*/g,`.*`).replace(/\?/g,`.`);return RegExp(`^${t}$`,`i`)}};Fr=V([v()],Fr);let Ir=class{constructor(e){this.bundler=e}async extractSpecMetadata(e){let t=await this.bundler.bundle(e);try{let e=await import(`${t.outputPath}?t=${Date.now()}`),n=e.argsSchema??null;return{argsSchema:n,envPrefix:e.envPrefix??null,hasDynamicArgs:n!==null}}catch{return{argsSchema:null,envPrefix:null,hasDynamicArgs:!1}}finally{await t.cleanup()}}async loadPlaywrightConfig(e){try{let t=await j(e,`utf-8`),n={},r=t.match(/baseURL:\s*['"`]([^'"`]+)['"`]/);r&&(n.baseURL=r[1]);let i=t.match(/export\s+const\s+mcpConfig\s*=\s*\{([^}]+)\}/s);if(i){let e=i[1],t=e.match(/setupFile:\s*['"`]([^'"`]+)['"`]/);t&&(n.setupFile=t[1]);let r=e.match(/setupExport:\s*['"`]([^'"`]+)['"`]/);r&&(n.setupExport=r[1])}let a=t.match(/webServer:\s*\{([^}]+(?:\{[^}]*\}[^}]*)*)\}/s);if(a){let e=a[1],t=e.match(/command:\s*['"`]([^'"`]+)['"`]/),r=e.match(/url:\s*['"`]([^'"`]+)['"`]/),i=e.match(/reuseExistingServer:\s*(true|false)/),o=e.match(/timeout:\s*(\d+)/),s=e.match(/cwd:\s*['"`]([^'"`]+)['"`]/);t&&r&&(n.webServer={command:t[1],url:r[1],reuseExistingServer:i?i[1]===`true`:void 0,timeout:o?Number.parseInt(o[1],10):void 0,cwd:s?s[1]:void 0})}return n}catch{return{}}}parseArgsFromEnv(e,t){let n={},r=t.toUpperCase();for(let[e,t]of Object.entries(process.env))if(e.startsWith(r)&&t!==void 0){let i=this.envKeyToFieldName(e.slice(r.length));n[i]=this.parseEnvValue(t)}try{return e.parse(n)}catch{return n}}envKeyToFieldName(e){return e.toLowerCase().split(`_`).map((e,t)=>t===0?e:e.charAt(0).toUpperCase()+e.slice(1)).join(``)}parseEnvValue(e){if(e.toLowerCase()===`true`)return!0;if(e.toLowerCase()===`false`)return!1;let t=Number(e);if(!Number.isNaN(t)&&e.trim()!==``)return t;if(e.startsWith(`{`)&&e.endsWith(`}`)||e.startsWith(`[`)&&e.endsWith(`]`))try{return JSON.parse(e)}catch{}return e}};Ir=V([v(),B(0,_(R.SpecBundlerService)),z(`design:paramtypes`,[Object])],Ir);function Lr(e,t){let{runtimeUse:r,unsupportedKeys:i}=n(t);return{options:{...e?{baseURL:e}:{},...r},warnings:i.length>0?[`test.use() ignores unsupported fixture keys: ${i.join(`, `)}`]:[]}}let Rr=class{constructor(e){this.bundler=e}async loadSpec(e,n){let a=n?await this.bundler.bundleWithHooks({specPath:e,hooksPath:n}):await this.bundler.bundle(e),o=process,s=o.__pw_initiator__;try{return t(e),o.__pw_initiator__=void 0,await import(`${a.outputPath}?t=${Date.now()}`),i()}catch(t){throw r(),Error(`Failed to load spec "${e}": ${t instanceof Error?t.message:String(t)}`,{cause:t})}finally{o.__pw_initiator__=s,await a.cleanup()}}async executeSpec(e){let{specPath:t,browserType:n=`chromium`,headless:r=!0,hooksPath:i,baseURL:o,keepBrowserOpen:s}=e,c=Date.now(),l=[],u=0,d=0,f=new Set,p=new Set,m=[],h=await be[n].launch({channel:`chrome`,headless:r});try{let e=await this.loadSpec(t,i);for(let t of e.allTests){let{options:e,warnings:n}=Lr(o,t.config.use);for(let e of n)f.add(e);let r=t.serialScopeId,i=r!==void 0&&p.has(r);if(t.skip||i){l.push({title:t.title,fullTitle:t.fullTitle,passed:!0,error:null,duration:0}),u++;continue}let c=await h.newContext(e),g=await c.newPage(),_={page:g,context:c,browser:h,playwright:be},v=Date.now();try{await t.fn(_),l.push({title:t.title,fullTitle:t.fullTitle,passed:!0,error:null,duration:Date.now()-v}),u++}catch(e){if(e instanceof a||e instanceof Error&&`isSkip`in e)l.push({title:t.title,fullTitle:t.fullTitle,passed:!0,error:null,duration:Date.now()-v}),u++;else{let n=e instanceof Error?e.message:String(e);l.push({title:t.title,fullTitle:t.fullTitle,passed:!1,error:n,duration:Date.now()-v}),d++,r!==void 0&&p.add(r)}}finally{s?m.push({page:g,context:c}):await c.close().catch(()=>{})}}if(!s){for(let e of h.contexts())await e.close().catch(()=>{});await h.close()}return{specPath:t,totalTests:e.testCount,passed:u,failed:d,testResults:l,success:d===0,duration:Date.now()-c,warnings:f.size>0?[...f]:void 0,browser:s?h:void 0,keptPages:m.length>0?m:void 0}}catch(e){for(let e of h.contexts())await e.close().catch(()=>{});return await h.close().catch(()=>{}),{specPath:t,totalTests:0,passed:0,failed:1,testResults:[{title:`Spec Loading`,fullTitle:`Failed to load: ${t}`,passed:!1,error:e instanceof Error?e.message:String(e),duration:Date.now()-c}],success:!1,duration:Date.now()-c,warnings:f.size>0?[...f]:void 0}}}async executeSpecEnhanced(e){let{specPath:t,browserType:n=`chromium`,headless:r=!0,testFilter:i,specArgs:o,hooksPath:s,baseURL:c,keepBrowserOpen:l}=e,u=Date.now(),d=[],f=0,p=0,m=new Set,h=new Set,g=[],_=await be[n].launch({channel:`chrome`,headless:r});try{let e=await this.loadSpec(t,s),n=i?this.filterTests(e.allTests,i):e.allTests;for(let e of n){let{options:t,warnings:n}=Lr(c,e.config.use);for(let e of n)m.add(e);let r=e.serialScopeId,i=r!==void 0&&h.has(r);if(e.skip||i){d.push({title:e.title,fullTitle:e.fullTitle,passed:!0,error:null,duration:0}),f++;continue}let s=await _.newContext(t),u=await s.newPage(),v={page:u,context:s,browser:_,playwright:be,specArgs:o},y=Date.now();try{await e.fn(v),d.push({title:e.title,fullTitle:e.fullTitle,passed:!0,error:null,duration:Date.now()-y}),f++}catch(t){if(t instanceof a||t instanceof Error&&`isSkip`in t)d.push({title:e.title,fullTitle:e.fullTitle,passed:!0,error:null,duration:Date.now()-y}),f++;else{let n=t instanceof Error?t.message:String(t);d.push({title:e.title,fullTitle:e.fullTitle,passed:!1,error:n,duration:Date.now()-y}),p++,r!==void 0&&h.add(r)}}finally{l?g.push({page:u,context:s}):await s.close().catch(()=>{})}}if(!l){for(let e of _.contexts())await e.close().catch(()=>{});await _.close()}return{specPath:t,totalTests:n.length,passed:f,failed:p,testResults:d,success:p===0,duration:Date.now()-u,warnings:m.size>0?[...m]:void 0,browser:l?_:void 0,keptPages:g.length>0?g:void 0}}catch(e){for(let e of _.contexts())await e.close().catch(()=>{});return await _.close().catch(()=>{}),{specPath:t,totalTests:0,passed:0,failed:1,testResults:[{title:`Spec Loading`,fullTitle:`Failed to load: ${t}`,passed:!1,error:e instanceof Error?e.message:String(e),duration:Date.now()-u}],success:!1,duration:Date.now()-u,warnings:m.size>0?[...m]:void 0}}}filterTests(e,t){let n=[...e];if(t.onlyMarked){let e=n.filter(e=>e.only);e.length>0&&(n=e)}if(t.testName&&(n=n.filter(e=>e.title===t.testName)),t.testPattern){let e=new RegExp(t.testPattern,`i`);n=n.filter(t=>e.test(t.fullTitle))}if(t.describeFilter){let e=new RegExp(t.describeFilter,`i`);n=n.filter(t=>e.test(t.fullTitle))}return n}};Rr=V([v(),B(0,_(R.SpecBundlerService)),z(`design:paramtypes`,[Object])],Rr);let zr=class{browsers=new Map;browserIdCounter=0;constructor(e,t,n){this.profileService=e,this.pageRegistry=t,this.lockManager=n}findChromePath(){let e=N.platform(),t={darwin:[`/Applications/Google Chrome.app/Contents/MacOS/Google Chrome`,`/Applications/Google Chrome Canary.app/Contents/MacOS/Google Chrome Canary`,`${N.homedir()}/Applications/Google Chrome.app/Contents/MacOS/Google Chrome`],win32:[`C:\\Program Files\\Google\\Chrome\\Application\\chrome.exe`,`C:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe`,`${process.env.LOCALAPPDATA}\\Google\\Chrome\\Application\\chrome.exe`],linux:[`/usr/bin/google-chrome`,`/usr/bin/google-chrome-stable`,`/usr/bin/chromium`,`/usr/bin/chromium-browser`,`/snap/bin/chromium`]},n=t[e]??t.linux;for(let e of n)if(O.existsSync(e))return e;return null}resolveExtensionPath(e){if(e&&O.existsSync(x.join(e,`manifest.json`)))return e;let t=x.dirname(new URL(import.meta.url).pathname),n=[x.resolve(t,`../dist/extension`),x.resolve(t,`../../dist/extension`)];for(let e of n)if(O.existsSync(x.join(e,`manifest.json`)))return e;let r=x.resolve(process.cwd(),`dist/extension`);if(O.existsSync(x.join(r,`manifest.json`)))return r;throw Error(`Chrome extension not found. Build the extension first: pnpm build:extension`)}createUserDataDir(e){if(e){let t=x.join(this.profileService.getProfilesDir(),e,`stealth-user-data`);return O.mkdirSync(t,{recursive:!0}),t}let t=x.join(N.tmpdir(),`stealth-chrome-${Date.now()}`);return O.mkdirSync(t,{recursive:!0}),t}async launch(e={}){let t=this.findChromePath();if(!t)throw Error(`Chrome not found. Please install Google Chrome.
|
|
13
|
+
macOS: brew install --cask google-chrome
|
|
14
|
+
Linux: sudo apt install google-chrome-stable`);let n=this.resolveExtensionPath(e.extensionPath),r=this.createUserDataDir(e.profileName);await this.lockManager.ensureAvailable(r);let i=`stealth-browser-${++this.browserIdCounter}`,a=[n];e.proxy?.username&&e.proxy?.password&&a.push(Rt(e.proxy.username,e.proxy.password));let o=a.join(`,`),s=F(t,[`--user-data-dir=${r}`,`--load-extension=${o}`,`--disable-extensions-except=${o}`,`--disable-blink-features=AutomationControlled`,`--disable-infobars`,`--no-first-run`,`--no-default-browser-check`,`--disable-background-timer-throttling`,`--disable-backgrounding-occluded-windows`,`--disable-renderer-backgrounding`,`--disable-component-update`,`--disable-features=TranslateUI`,`--disable-ipc-flooding-protection`,`--password-store=basic`,`--use-mock-keychain`,`--disable-dev-shm-usage`,`--disable-gpu-sandbox`,`--silent-debugger-extension-api`,...e.windowSize?[`--window-size=${e.windowSize.width},${e.windowSize.height}`]:[`--window-size=1280,720`],...e.proxy?zt(e.proxy):[],e.url??`about:blank`,...e.extraArgs??[]],{detached:!1,stdio:[`ignore`,`ignore`,`ignore`],env:{...process.env,DISPLAY:process.env.DISPLAY}});if(!s.pid)throw Error(`Failed to start Chrome process`);let c={id:i,process:s,pid:s.pid,userDataDir:r,extensionPath:n,profileName:e.profileName,createdAt:new Date,isRunning:!0};return s.on(`exit`,()=>{c.isRunning=!1}),s.on(`error`,e=>{console.error(`Chrome process ${i} error:`,e),c.isRunning=!1}),this.browsers.set(i,c),this.pageRegistry.registerExtensionPage(i),c}async close(e){let t=this.browsers.get(e);if(t&&(t.isRunning&&t.pid&&await Lt(t.pid),t.isRunning=!1,this.browsers.delete(e),!t.profileName&&t.userDataDir.includes(`stealth-chrome-`)))try{O.rmSync(t.userDataDir,{recursive:!0,force:!0})}catch{}}async closeAll(){let e=Array.from(this.browsers.keys()).map(e=>this.close(e));await Promise.all(e)}getBrowser(e){return this.browsers.get(e)}listBrowsers(){return Array.from(this.browsers.values()).filter(e=>e.isRunning)}};zr=V([v(),B(0,_(R.ProfileService)),B(1,_(R.PageRegistry)),B(2,_(R.BrowserLockManager)),z(`design:paramtypes`,[Object,Object,Object])],zr);function Br(e){if(!e.isError)return;let t=e.content[0];return t?.type===`text`&&typeof t.text==`string`?t.text:`Unknown tool execution error`}let Vr=class{constructor(e,t,n,r=new G){this.pageRegistry=e,this.browserService=t,this.delegator=n,this.telemetry=r}async execute(e,t,n,r){return this.telemetry.runInSpan(`browse_tool.tool_executor.page`,{attributes:{"browse_tool.tool.name":e,"browse_tool.page.id":t}},async i=>{let a=this.pageRegistry.get(t);if(!a)return i?.setStatus({code:ke.ERROR,message:`Page "${t}" not found`}),{content:[{type:`text`,text:`Page "${t}" not found`}],isError:!0};i?.setAttributes({"browse_tool.browser.id":a.browserId,"browse_tool.execution.mode":a.mode});let o=a.mode===`extension`?await this.delegator.executeTool(e,{pageId:t,browserId:a.browserId,...n}):await r(),s=Br(o);if(s)i?.setStatus({code:ke.ERROR,message:s});else if(a.mode===`extension`)try{await this.browserService.ensureExtensionRecordingActive(a.browserId,t)}catch(e){console.warn(`[ToolExecutor] Failed to start extension recording for browser "${a.browserId}" on page "${t}":`,e)}return o})}isExtensionMode(e){return this.pageRegistry.get(e)?.mode===`extension`}async executeForBrowser(e,t,n,r){return this.telemetry.runInSpan(`browse_tool.tool_executor.browser`,{attributes:{"browse_tool.tool.name":e,"browse_tool.browser.id":t}},async i=>{let a=this.browserService.getBrowser(t);if(!a)return i?.setStatus({code:ke.ERROR,message:`Browser "${t}" not found`}),{content:[{type:`text`,text:`Browser "${t}" not found`}],isError:!0};i?.setAttributes({"browse_tool.execution.mode":a.mode});let o=a.mode===`extension`||a.mode===`vm`?await this.delegator.executeTool(e,{browserId:t,...n}):await r(),s=Br(o);return s&&i?.setStatus({code:ke.ERROR,message:s}),o})}isBrowserExtensionMode(e){let t=this.browserService.getBrowser(e);return t?.mode===`extension`||t?.mode===`vm`}};Vr=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ExtensionToolDelegator)),B(3,_(R.TelemetryService)),z(`design:paramtypes`,[Object,Object,Object,Object])],Vr);let Hr=class{serverProcess=null;currentUrl=null;cleanupRegistered=!1;constructor(){this.registerCleanup()}registerCleanup(){if(this.cleanupRegistered)return;this.cleanupRegistered=!0;let e=()=>this.cleanupSync();process.on(`exit`,e),process.on(`SIGINT`,e),process.on(`SIGTERM`,e)}async startServer(e,t){let{command:n,url:r,reuseExistingServer:i=!1,timeout:a=3e4,cwd:o}=e;if(i&&await this.isServerRunning(r))return this.currentUrl=r,{started:!1,url:r,reused:!0};await this.stopServer();let s=o?D(t,o):t,[c,...l]=this.parseCommand(n);if(this.serverProcess=F(c,l,{cwd:s,shell:!0,stdio:[`ignore`,`pipe`,`pipe`],detached:!1}),this.currentUrl=r,this.serverProcess.on(`error`,e=>{console.error(`Server process error: ${e.message}`)}),!await this.waitForServer(r,a))throw await this.stopServer(),Error(`Server failed to start within ${a}ms. URL: ${r}, Command: ${n}`);return{started:!0,url:r,reused:!1}}async waitForServer(e,t){let n=Date.now(),r=100;for(;Date.now()-n<t;){if(await this.isServerRunning(e))return!0;await this.sleep(r),r=Math.min(r*1.5,2e3)}return!1}async stopServer(){if(this.serverProcess)return new Promise(e=>{if(!this.serverProcess){e();return}let t=this.serverProcess;this.serverProcess=null,this.currentUrl=null;let n=setTimeout(()=>{try{t.kill(`SIGKILL`)}catch{}e()},5e3);t.on(`exit`,()=>{clearTimeout(n),e()});try{t.kill(`SIGTERM`)}catch{clearTimeout(n),e()}})}async isServerRunning(e){try{let t=new AbortController,n=setTimeout(()=>t.abort(),3e3),r=await fetch(e,{method:`HEAD`,signal:t.signal});return clearTimeout(n),r.status>=200&&r.status<400}catch{return!1}}parseCommand(e){let t=[],n=``,r=!1,i=``;for(let a of e)(a===`"`||a===`'`)&&!r?(r=!0,i=a):a===i&&r?(r=!1,i=``):a===` `&&!r?n&&=(t.push(n),``):n+=a;return n&&t.push(n),t}sleep(e){return new Promise(t=>setTimeout(t,e))}getCurrentUrl(){return this.currentUrl}cleanupSync(){if(this.serverProcess){try{this.serverProcess.kill(`SIGKILL`)}catch{}this.serverProcess=null,this.currentUrl=null}}};Hr=V([v(),z(`design:paramtypes`,[])],Hr);const Ur=I.object({type:I.string(),text:I.string().optional(),data:I.string().optional(),mimeType:I.string().optional()}),Wr=I.object({traceId:I.string().regex(/^[0-9a-f]{32}$/i).optional(),parentSpanId:I.string().regex(/^[0-9a-f]{16}$/i).optional()}),Gr=I.object({content:I.array(Ur),isError:I.boolean().optional()}),Kr=I.object({type:I.literal(`task:push`),id:I.string(),payload:I.object({taskId:I.string(),tool:I.string(),arguments:I.record(I.string(),I.unknown()),pageId:I.string().optional(),telemetry:Wr.optional()})}),qr=I.object({type:I.literal(`page:created`),payload:I.object({pageId:I.string(),browserId:I.string(),url:I.string().optional()})}),Jr=I.object({type:I.literal(`page:removed`),payload:I.object({pageId:I.string()})}),Yr=I.object({type:I.literal(`session:ack`),id:I.string(),payload:I.object({sessionId:I.string(),controlMode:I.string()})}),Xr=I.object({type:I.literal(`pong`)}),Zr=I.object({type:I.literal(`error`),payload:I.object({message:I.string(),code:I.string().optional()})});I.discriminatedUnion(`type`,[Kr,qr,Jr,Yr,Xr,Zr]);const Qr=I.object({type:I.literal(`session:register`),id:I.string(),payload:I.object({browserId:I.string(),tabId:I.number().optional(),url:I.string().optional()})}),$r=I.object({type:I.literal(`task:result`),payload:I.object({taskId:I.string(),success:I.boolean(),result:Gr.optional(),error:I.string().optional()})}),ei=I.object({type:I.literal(`tab:mapped`),payload:I.object({pageId:I.string(),tabId:I.number()})}),ti=I.object({type:I.literal(`ping`)}),ni=I.object({type:I.literal(`heartbeat`),payload:I.object({sessionId:I.string(),tabId:I.number().optional(),url:I.string().optional()})}),ri=I.discriminatedUnion(`type`,[Qr,$r,ei,ti,ni]);function ii(e){let t=ri.safeParse(e);return t.success?t.data:null}let ai=class{connections=new Map;browserConnections=new Map;eventHandlers={};maxConnections=100;setEventHandlers(e){this.eventHandlers=e}addConnection(e,t){if(this.connections.size>=this.maxConnections)throw Error(`Maximum connections (${this.maxConnections}) reached`);let n=crypto.randomUUID(),r={id:n,ws:e,browserId:t,connectedAt:new Date,lastMessageAt:new Date};return this.connections.set(n,r),this.browserConnections.has(t)||this.browserConnections.set(t,new Set),this.browserConnections.get(t).add(n),console.log(`[WebSocketHub] Connection added: ${n} for browser ${t}`),n}removeConnection(e){let t=this.connections.get(e);if(!t)return;this.eventHandlers.onDisconnect?.(t);let n=this.browserConnections.get(t.browserId);n&&(n.delete(e),n.size===0&&this.browserConnections.delete(t.browserId)),this.connections.delete(e),console.log(`[WebSocketHub] Connection removed: ${e}`)}getConnection(e){return this.connections.get(e)}getConnectionsByBrowser(e){let t=this.browserConnections.get(e);return t?Array.from(t).map(e=>this.connections.get(e)).filter(e=>e!==void 0):[]}hasConnection(e){return this.browserConnections.has(e)&&(this.browserConnections.get(e)?.size??0)>0}reassignConnection(e,t){let n=this.connections.get(e);if(!n)return!1;let r=n.browserId;if(r===t)return!0;let i=this.browserConnections.get(r);return i&&(i.delete(e),i.size===0&&this.browserConnections.delete(r)),this.browserConnections.has(t)||this.browserConnections.set(t,new Set),this.browserConnections.get(t).add(e),n.browserId=t,!0}handleMessage(e,t){let n=this.connections.get(e);if(!n){console.warn(`[WebSocketHub] Message from unknown connection: ${e}`);return}n.lastMessageAt=new Date;try{let r=typeof t==`string`?t:new TextDecoder().decode(t),i=JSON.parse(r),a=ii(i);if(!a){console.warn(`[WebSocketHub] Invalid message format from ${e}:`,i),this.sendError(n,`Invalid message format`);return}this.routeMessage(n,a)}catch(t){console.error(`[WebSocketHub] Error parsing message from ${e}:`,t),this.sendError(n,`Failed to parse message`)}}routeMessage(e,t){switch(t.type){case`session:register`:this.eventHandlers.onSessionRegister?.(e,t);break;case`task:result`:this.eventHandlers.onTaskResult?.(e,t);break;case`tab:mapped`:this.eventHandlers.onTabMapped?.(e,t);break;case`heartbeat`:this.eventHandlers.onHeartbeat?.(e,t);break;case`ping`:this.sendToConnection(e,{type:`pong`});break}}sendToConnection(e,t){try{return e.ws.send(JSON.stringify(t)),!0}catch(t){return console.error(`[WebSocketHub] Failed to send to ${e.id}:`,t),!1}}sendError(e,t,n){this.sendToConnection(e,{type:`error`,payload:{message:t,code:n}})}pushTask(e,t){let n=this.getConnectionsByBrowser(e);if(n.length===0)return console.warn(`[WebSocketHub] No connections for browser ${e}`),!1;let r=n[0];return this.sendToConnection(r,t)}broadcastPageCreated(e,t,n){let r={type:`page:created`,payload:{pageId:t,browserId:e,url:n}},i=this.getConnectionsByBrowser(e);for(let e of i)this.sendToConnection(e,r)}broadcastPageRemoved(e,t){let n={type:`page:removed`,payload:{pageId:t}},r=this.getConnectionsByBrowser(e);for(let e of r)this.sendToConnection(e,n)}sendSessionAck(e,t,n,r){e.sessionId=n;let i={type:`session:ack`,id:t,payload:{sessionId:n,controlMode:r}};this.sendToConnection(e,i)}getStats(){return{totalConnections:this.connections.size,browserCount:this.browserConnections.size,connections:Array.from(this.connections.values()).map(e=>({id:e.id,browserId:e.browserId,connectedAt:e.connectedAt}))}}closeAll(){for(let[e,t]of this.connections){try{t.ws.close()}catch{}this.removeConnection(e)}}};ai=V([v()],ai);let $=class{constructor(e,t,n){this.pageRegistry=e,this.browserService=t,this.toolExecutor=n}getPage(e){return this.pageRegistry.get(e)}resolvePage(e){let t=this.pageRegistry.get(e);if(!t)throw Error(`Page "${e}" not found`);if(!t.page)throw Error(`Page "${e}" is in extension mode and has no Playwright page`);return t.page}resolvePageEntry(e){let t=this.pageRegistry.get(e);if(!t)throw Error(`Page "${e}" not found`);return t}success(e){return{content:[{type:`text`,text:e}]}}successJson(e){return{content:[{type:`text`,text:JSON.stringify(e,null,2)}]}}successImage(e,t=`image/png`){return{content:[{type:`image`,data:e,mimeType:t}]}}error(e){return{content:[{type:`text`,text:e}],isError:!0}}async safeExecute(e){try{return await e()}catch(e){let t=e instanceof Error?e.message:String(e);return this.error(t)}}async executeWithMode(e,t,n,r){return this.toolExecutor.execute(e,t,n,r)}};$=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],$);var oi;const si=W.extend({pageId:I.string().describe(`Page ID to operate on`),limit:I.number().int().min(1).max(10).optional().default(5).describe(`Maximum number of candidates to return`)});async function ci(e,t,n=5){return(await ui(e,t.frame)).evaluate(({selector:e,limit:t})=>{let n=e=>(e??``).replace(/\s+/g,` `).trim(),r=(e,t,r=!1)=>{if(!t)return!1;let i=n(e),a=n(t);return!i||!a?!1:r?i===a:i.toLowerCase().includes(a.toLowerCase())},i=e=>{if(e instanceof HTMLInputElement||e instanceof HTMLTextAreaElement||e instanceof HTMLSelectElement){if(typeof e.labels?.length==`number`&&e.labels.length>0)return n(e.labels[0]?.textContent);if(e.id){let t=document.querySelector(`label[for="${CSS.escape(e.id)}"]`);if(t?.textContent)return n(t.textContent)}}let t=e.closest(`label`);return t?.textContent?n(t.textContent):null},a=e=>{let t=e.getAttribute(`aria-label`);if(t)return n(t);let r=e.getAttribute(`aria-labelledby`);if(r){let e=r.split(/\s+/).map(e=>n(document.getElementById(e)?.textContent)).filter(Boolean);if(e.length>0)return e.join(` `)}return i(e)||(e instanceof HTMLInputElement&&e.placeholder?n(e.placeholder):n(e.textContent))},o=e=>{let t=e.getAttribute(`role`);if(t)return t;if(e instanceof HTMLInputElement)return{button:`button`,checkbox:`checkbox`,email:`textbox`,number:`spinbutton`,password:`textbox`,radio:`radio`,range:`slider`,search:`searchbox`,submit:`button`,tel:`textbox`,text:`textbox`,url:`textbox`}[(e.type||`text`).toLowerCase()]||`textbox`;let n=e.tagName.toLowerCase();return{a:e.hasAttribute(`href`)?`link`:`none`,button:`button`,img:`img`,option:`option`,select:`combobox`,textarea:`textbox`}[n]||`none`},s=new Set([`html`,`body`,`head`,`script`,`style`,`noscript`,`template`,`meta`,`link`]),c=t=>{if(s.has(t.tagName.toLowerCase()))return null;let c=e.exact??!1,l=a(t),u=i(t),d=t.getAttribute(`placeholder`),f=n(t.textContent),p=o(t),m=t.getAttribute(`data-testid`)||t.getAttribute(`data-test-id`),h=t.getBoundingClientRect(),g=h.width>0&&h.height>0,_=!(t instanceof HTMLButtonElement||t instanceof HTMLInputElement||t instanceof HTMLSelectElement||t instanceof HTMLTextAreaElement)||!t.disabled,v=0,y=0,b=[],x=p!==`none`||t instanceof HTMLButtonElement||t instanceof HTMLInputElement||t instanceof HTMLSelectElement||t instanceof HTMLTextAreaElement||t instanceof HTMLAnchorElement&&t.hasAttribute(`href`);return e.role&&p===e.role&&(v+=30,y+=1,b.push(`role match`)),e.name&&(r(l,e.name,!0)?(v+=60,y+=1,b.push(`exact accessible name match`)):r(l,e.name,c)&&(v+=35,y+=1,b.push(`accessible name match`))),e.label&&(r(u,e.label,!0)?(v+=55,y+=1,b.push(`exact label match`)):r(u,e.label,c)&&(v+=30,y+=1,b.push(`label match`))),e.placeholder&&r(d,e.placeholder,c)&&(v+=25,y+=1,b.push(`placeholder match`)),e.text&&r(f,e.text,c)&&(v+=20,y+=1,b.push(`text match`)),e.testId&&m===e.testId&&(v+=50,y+=1,b.push(`test id match`)),x&&(v+=8,b.push(`interactive element`)),g&&(v+=10,b.push(`visible`)),_&&(v+=5,b.push(`enabled`)),y===0?null:{score:v,reasons:b,candidate:{tagName:t.tagName.toLowerCase(),id:t.id||``,className:t.getAttribute(`class`)||``,role:p,accessibleName:l,label:u,placeholder:d,testId:m,textContent:f.slice(0,500),visible:g,enabled:_,rect:{x:h.x,y:h.y,width:h.width,height:h.height}}}},l=Array.from(document.querySelectorAll(`*`)).map(e=>c(e)).filter(e=>e!==null&&e.score>0).sort((e,t)=>t.score-e.score).slice(0,t);return{totalCandidates:l.length,candidates:l}},{selector:t,limit:n})}let li=class extends ${static{oi=this}static TOOL_NAME=`browser_resolve_locator`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return si}getDefinition(){return{name:oi.TOOL_NAME,description:`Returns ranked locator candidates for a selector intent. Use this after a failed click/fill to recover using semantic matches instead of brittle guesses.`,inputSchema:I.toJSONSchema(si,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>this.executeWithMode(oi.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=this.buildSelector(e),r=await ci(t,n,e.limit??5);return this.successJson({pageId:e.pageId,selector:n,...r})}))}buildSelector(e){return{uid:e.uid,role:e.role,name:e.name,label:e.label,placeholder:e.placeholder,testId:e.testId,text:e.text,exact:e.exact,selector:e.selector,xpath:e.xpath,frame:e.frame}}};li=oi=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],li);async function ui(e,t){if(!t)return e;let n=await(await e.locator(t).first().elementHandle())?.contentFrame();if(!n)throw Error(`Frame not found: ${t}`);return n}var di;const fi=W.extend({pageId:I.string().describe(`Optional page ID to override browser current page`),clickCount:I.number().optional().default(1).describe(`Number of clicks (default: 1, use 2 for double-click)`),button:I.enum([`left`,`middle`,`right`]).optional().default(`left`).describe(`Mouse button to use`),modifiers:I.array(I.enum([`Alt`,`Control`,`Meta`,`Shift`])).optional().describe(`Modifier keys to hold during click`),delay:I.number().optional().describe(`Time to wait between mousedown and mouseup in ms`),position:I.object({x:I.number(),y:I.number()}).optional().describe(`Position offset within the element`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds (default: ${H})`)});let pi=class extends ${static{di=this}static TOOL_NAME=`browser_click`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return fi}getDefinition(){return{name:di.TOOL_NAME,description:`Clicks on an element on the page. Supports CSS selector, XPath, text content, or accessibility snapshot UID for element identification.`,inputSchema:I.toJSONSchema(fi,{reused:`inline`})}}async execute(e){return this.executeWithMode(di.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=this.buildSelector(e);try{return await(await this.elementLocator.locate(t,n)).click({clickCount:e.clickCount,button:e.button,modifiers:e.modifiers,delay:e.delay,position:e.position,timeout:e.timeout}),this.success(`Clicked element successfully`)}catch(r){let i=r instanceof Error?r.message:String(r);if(!i.includes(`Element not found`))throw r;let a=await ci(t,n,5);return this.error(JSON.stringify({error:i,pageId:e.pageId,selector:n,...a},null,2))}})}buildSelector(e){return{uid:e.uid,role:e.role,name:e.name,label:e.label,placeholder:e.placeholder,testId:e.testId,text:e.text,exact:e.exact,selector:e.selector,xpath:e.xpath,frame:e.frame}}};pi=di=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],pi);var mi;const hi=I.object({browserId:I.string().describe(`The browser ID to close (returned by browser_launch)`)});let gi=class extends ${static{mi=this}static TOOL_NAME=`browser_close`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return hi}getDefinition(){return{name:mi.TOOL_NAME,description:`Closes a browser instance by its ID. Terminates the browser process, cleans up all associated pages, and releases resources. Works for all modes: playwright, extension, vm (Docker), and stealth.`,inputSchema:I.toJSONSchema(hi,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.browserService.getBrowser(e.browserId);if(!t)return this.error(`Browser "${e.browserId}" not found`);let n=t.pageIds.size,r=t.mode??`playwright`;return await this.browserService.closeBrowser(e.browserId),this.successJson({closedBrowserId:e.browserId,mode:r,pagesRemoved:n,message:`Browser "${e.browserId}" closed successfully`})})}};gi=mi=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],gi);var _i;const vi=I.object({pageId:I.string().describe(`Page ID to close`)});let yi=class extends ${static{_i=this}static TOOL_NAME=`browser_close_page`;constructor(e,t,n,r){super(e,t,n),this.profileService=r}getInputSchema(){return vi}getDefinition(){return{name:_i.TOOL_NAME,description:`Closes a page (tab) by its ID. Removes the page from the session and registry. If closed page was current, another page becomes current.`,inputSchema:I.toJSONSchema(vi,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.pageRegistry.get(e.pageId);if(!t)return this.error(`Page "${e.pageId}" not found`);let{browserId:n,profileName:r,mode:i}=t,a=this.browserService.getBrowser(n);if(!a)return this.error(`Browser "${n}" not found`);let o=a.currentPageId===e.pageId,s=a.pageIds.size===1;if(i===`extension`||i===`vm`){let t=await this.executeWithMode(_i.TOOL_NAME,e.pageId,e,async()=>this.error(`Page "${e.pageId}" is not in extension mode`));if(t.isError)return t;if(await this.browserService.finalizeExtensionRecording(n,e.pageId),this.pageRegistry.remove(e.pageId),a.pageIds.delete(e.pageId),a.currentPageId===e.pageId&&(a.currentPageId=Array.from(a.pageIds)[0]??null),s)try{await this.browserService.closeBrowser(n)}catch(e){console.error(`Failed to close extension browser "${n}" after last tab shutdown:`,e)}return this.successJson({closedPageId:e.pageId,browserId:n,wasCurrentPage:o,newCurrentPageId:a.currentPageId,browserKeptOpen:!s,message:s?`Page "${e.pageId}" closed. Browser "${n}" was also closed because no tabs remained.`:`Page "${e.pageId}" closed successfully`})}if(!t.page)return this.error(`Page "${e.pageId}" is in extension mode and cannot be closed via Playwright`);await t.page.close();let c=this.browserService.getBrowser(n)?.currentPageId??null;if(s){if(r&&a.context)try{let e=await a.context.storageState();await this.profileService.saveStorageState(r,e)}catch(e){console.error(`Failed to auto-save profile "${r}":`,e)}if(!r)try{await this.browserService.closeBrowser(n)}catch(e){console.error(`Failed to auto-close browser "${n}":`,e)}}return this.successJson({closedPageId:e.pageId,browserId:n,wasCurrentPage:o,newCurrentPageId:c,browserKeptOpen:s&&!!r,message:s&&r?`Page "${e.pageId}" closed. Browser kept open for profile "${r}" reuse.`:`Page "${e.pageId}" closed successfully`})})}};yi=_i=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ProfileService)),z(`design:paramtypes`,[Object,Object,Object,Object])],yi);var bi;const xi=I.object({name:I.string().describe(`Unique profile name`),browserType:I.enum([`chromium`,`firefox`,`webkit`]).optional().default(`chromium`).describe(`Preferred browser type for the profile`),viewport:I.object({width:I.number(),height:I.number()}).optional(),userAgent:I.string().optional(),locale:I.string().optional(),timezone:I.string().optional(),colorScheme:I.enum([`light`,`dark`,`no-preference`]).optional()});let Si=class{static{bi=this}static TOOL_NAME=`browser_create_profile`;constructor(e){this.profileService=e}getInputSchema(){return xi}getDefinition(){return{name:bi.TOOL_NAME,description:`Creates a browser profile for persistent browser settings and session reuse.`,inputSchema:I.toJSONSchema(xi,{reused:`inline`})}}async execute(e){try{let t=await this.profileService.create({name:e.name,browserType:e.browserType??`chromium`,viewport:e.viewport,userAgent:e.userAgent,locale:e.locale,timezone:e.timezone,colorScheme:e.colorScheme});return{content:[{type:`text`,text:JSON.stringify(t,null,2)}]}}catch(e){return{content:[{type:`text`,text:e instanceof Error?e.message:String(e)}],isError:!0}}}};Si=bi=V([v(),B(0,_(R.ProfileService)),z(`design:paramtypes`,[Object])],Si);var Ci;const wi=I.object({name:I.string().describe(`Name of the profile to delete`)});let Ti=class{static{Ci=this}static TOOL_NAME=`browser_delete_profile`;constructor(e){this.profileService=e}getInputSchema(){return wi}getDefinition(){return{name:Ci.TOOL_NAME,description:`Deletes a browser profile and all its associated storage state data.`,inputSchema:I.toJSONSchema(wi,{reused:`inline`})}}async execute(e){try{return await this.profileService.delete(e.name),{content:[{type:`text`,text:JSON.stringify({success:!0,message:`Profile "${e.name}" deleted successfully`},null,2)}]}}catch(e){return{content:[{type:`text`,text:`Error: ${e instanceof Error?e.message:`Unknown error`}`}],isError:!0}}}};Ti=Ci=V([v(),B(0,_(R.ProfileService)),z(`design:paramtypes`,[Object])],Ti);var Ei;const Di=I.object({basePath:I.string().describe(`Absolute path to search for e2e projects (required)`),projectName:I.string().optional().describe(`Filter by project name (e.g., 'boomlink-e2e')`),specPattern:I.string().optional().describe(`Filter specs by name pattern (supports * and ? wildcards, e.g., "auth*", "*.spec.ts")`),includeMetadata:I.boolean().optional().default(!0).describe(`Include metadata like hasArgsSchema detection (default: true)`)});let Oi=class{static{Ei=this}static TOOL_NAME=`discover_specs`;constructor(e){this.specDiscoveryService=e}getInputSchema(){return Di}getDefinition(){return{name:Ei.TOOL_NAME,description:`Discovers e2e projects (directories with playwright.config.ts) and their spec files. Use to find available specs before running them.`,inputSchema:I.toJSONSchema(Di,{reused:`inline`})}}async execute(e){if(!e.basePath||!w(e.basePath))return{content:[{type:`text`,text:`Error: basePath must be an absolute path (e.g., /Users/me/project)`}],isError:!0};try{let t=e.basePath,n=e.includeMetadata!==!1;if(e.projectName||e.specPattern){let r=await this.specDiscoveryService.filterSpecs(t,{projectName:e.projectName,specPattern:e.specPattern}),i={filter:{basePath:t,projectName:e.projectName,specPattern:e.specPattern},totalSpecs:r.length,specs:r.map(e=>({name:e.name,path:e.path,relativePath:e.relativePath,...n&&{hasArgsSchema:e.hasArgsSchema}}))};return{content:[{type:`text`,text:JSON.stringify(i,null,2)}]}}let r=await this.specDiscoveryService.discoverProjects(t),i={basePath:t,totalProjects:r.length,totalSpecs:r.reduce((e,t)=>e+t.specs.length,0),projects:r.map(e=>({name:e.name,path:e.path,configPath:e.configPath,testDir:e.testDir,specCount:e.specs.length,specs:e.specs.map(e=>({name:e.name,path:e.path,relativePath:e.relativePath,...n&&{hasArgsSchema:e.hasArgsSchema}}))}))};return{content:[{type:`text`,text:JSON.stringify(i,null,2)}]}}catch(e){return{content:[{type:`text`,text:`Error discovering specs: ${e instanceof Error?e.message:`Unknown error`}`}],isError:!0}}}};Oi=Ei=V([v(),B(0,_(R.SpecDiscoveryService)),z(`design:paramtypes`,[Object])],Oi);var ki;const Ai=W.extend({x:I.number().optional().describe(`X coordinate`),y:I.number().optional().describe(`Y coordinate`)}),ji=I.object({pageId:I.string().describe(`Page ID to operate on`),source:W.describe(`Source element selector`),target:Ai.describe(`Target element selector or coordinates`),force:I.boolean().optional().default(!1).describe(`Bypass actionability checks`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let Mi=class extends ${static{ki=this}static TOOL_NAME=`browser_drag`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return ji}getDefinition(){return{name:ki.TOOL_NAME,description:`Drags an element to a target location or element.`,inputSchema:I.toJSONSchema(ji,{reused:`inline`})}}async execute(e){return this.executeWithMode(ki.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=await this.elementLocator.locate(t,e.source);if(`x`in e.target&&`y`in e.target)await n.dragTo(t.locator(`body`),{targetPosition:{x:e.target.x,y:e.target.y},force:e.force,timeout:e.timeout});else{let r=await this.elementLocator.locate(t,e.target);await n.dragTo(r,{force:e.force,timeout:e.timeout})}return this.success(`Dragged element successfully`)})}};Mi=ki=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],Mi);var Ni;const Pi=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),geolocation:I.object({latitude:I.number().min(-90).max(90),longitude:I.number().min(-180).max(180),accuracy:I.number().min(0).optional()}).optional().describe(`Geolocation coordinates to emulate`),locale:I.string().optional().describe(`Locale to emulate (e.g., "en-US", "de-DE")`),timezone:I.string().optional().describe(`Timezone to emulate (e.g., "America/New_York", "Europe/London")`),userAgent:I.string().optional().describe(`User agent string to use`),offline:I.boolean().optional().describe(`Whether to emulate offline mode`),colorScheme:I.enum([`light`,`dark`,`no-preference`]).optional().describe(`Color scheme preference`),reducedMotion:I.enum([`reduce`,`no-preference`]).optional().describe(`Reduced motion preference`)});let Fi=class extends ${static{Ni=this}static TOOL_NAME=`browser_emulate`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Pi}getDefinition(){return{name:Ni.TOOL_NAME,description:`Emulates device characteristics like geolocation, timezone, locale, color scheme, and network conditions.`,inputSchema:I.toJSONSchema(Pi,{reused:`inline`})}}async execute(e){return this.executeWithMode(Ni.TOOL_NAME,e.pageId,e,async()=>this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId),n=t.context,r=t.page;if(!n||!r)return this.error(`Page "${e.pageId}" is in extension mode and cannot be emulated via Playwright`);let i={};return e.geolocation&&(await n.setGeolocation(e.geolocation),i.geolocation=e.geolocation),e.offline!==void 0&&(await n.setOffline(e.offline),i.offline=e.offline),e.colorScheme&&(await r.emulateMedia({colorScheme:e.colorScheme}),i.colorScheme=e.colorScheme),e.reducedMotion&&(await r.emulateMedia({reducedMotion:e.reducedMotion}),i.reducedMotion=e.reducedMotion),e.locale&&(i.locale=e.locale,i.localeNote=`Locale changes may require a new browser context to take full effect`),e.timezone&&(i.timezone=e.timezone,i.timezoneNote=`Timezone changes may require a new browser context to take full effect`),e.userAgent&&(i.userAgent=e.userAgent,i.userAgentNote=`User agent changes may require a new browser context to take full effect`),this.successJson({success:!0,pageId:t.id,appliedSettings:i})}))}};Fi=Ni=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Fi);var Ii;const Li=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),script:I.string().describe(`JavaScript code to execute. Can be a function body or expression (e.g., "document.title" or "() => window.location.href")`),arg:I.unknown().optional().describe(`Optional argument to pass to the script if it is a function`)});let Ri=class extends ${static{Ii=this}static TOOL_NAME=`browser_evaluate_script`;static INLINE_SIZE_THRESHOLD=1e4;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Li}getDefinition(){return{name:Ii.TOOL_NAME,description:`Executes JavaScript code in the page context and returns the result. The script can be a function body or an expression.`,inputSchema:I.toJSONSchema(Li,{reused:`inline`})}}async execute(e){return this.executeWithMode(Ii.TOOL_NAME,e.pageId,{script:e.script,arg:e.arg},async()=>{let t=await this.resolvePage(e.pageId).evaluate(e.script,e.arg),n;try{n=JSON.stringify(t,null,2)}catch{n=String(t)}if(Buffer.byteLength(n,`utf-8`)<=Ii.INLINE_SIZE_THRESHOLD)return this.success(n);let r=`evaluate-${L()}.json`,i=T(P(),r);return await M(i,n,`utf-8`),this.success(`Result saved to: ${i}`)})}};Ri=Ii=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Ri);var zi;const Bi=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),selector:I.string().optional().describe(`CSS selector for the element to assert on (not required for page-level assertions like toHaveTitle, toHaveURL)`),assertion:I.enum([`toBeVisible`,`toBeHidden`,`toBeEnabled`,`toBeDisabled`,`toBeChecked`,`toBeEditable`,`toBeFocused`,`toBeEmpty`,`toHaveText`,`toContainText`,`toHaveValue`,`toHaveAttribute`,`toHaveClass`,`toHaveCount`,`toHaveCSS`,`toHaveId`,`toHaveTitle`,`toHaveURL`]).describe(`Type of assertion to perform`),expected:I.union([I.string(),I.number()]).optional().describe(`Expected value for assertions that require one (e.g., toHaveText, toHaveValue)`),attributeName:I.string().optional().describe(`Attribute name for toHaveAttribute or toHaveCSS assertion`),not:I.boolean().optional().describe(`Whether to negate the assertion (not.toBeVisible, etc.)`),timeout:I.number().optional().describe(`Timeout in milliseconds for the assertion (default: 5000)`)});let Vi=class extends ${static{zi=this}static TOOL_NAME=`browser_expect`;constructor(e,t,n,r){super(e,t,n),this.extensionPageProxy=r}getInputSchema(){return Bi}getDefinition(){return{name:zi.TOOL_NAME,description:`Performs Playwright expect() assertions on page elements. Supports various assertion types like toBeVisible, toHaveText, toHaveValue, toBeEnabled, etc.`,inputSchema:I.toJSONSchema(Bi,{reused:`inline`})}}async execute(t){return this.safeExecute(async()=>{let n=this.resolvePageEntry(t.pageId),r=n.mode===`extension`?(this.extensionPageProxy.setTarget(n.id,n.browserId),this.extensionPageProxy):this.resolvePage(t.pageId),{selector:i,assertion:a,expected:o,attributeName:s,not:c,timeout:l=5e3}=t,u=[`toHaveTitle`,`toHaveURL`];if(!u.includes(a)&&!i)return this.error(`Selector is required for assertion type: ${a}`);try{if(u.includes(a)){let t=c?e(r).not:e(r);a===`toHaveTitle`?await t.toHaveTitle(o,{timeout:l}):a===`toHaveURL`&&await t.toHaveURL(o,{timeout:l})}else{let t=r.locator(i),n=c?e(t).not:e(t);switch(a){case`toBeVisible`:await n.toBeVisible({timeout:l});break;case`toBeHidden`:await n.toBeHidden({timeout:l});break;case`toBeEnabled`:await n.toBeEnabled({timeout:l});break;case`toBeDisabled`:await n.toBeDisabled({timeout:l});break;case`toBeChecked`:await n.toBeChecked({timeout:l});break;case`toBeEditable`:await n.toBeEditable({timeout:l});break;case`toBeFocused`:await n.toBeFocused({timeout:l});break;case`toBeEmpty`:await n.toBeEmpty({timeout:l});break;case`toHaveText`:await n.toHaveText(o,{timeout:l});break;case`toContainText`:await n.toContainText(o,{timeout:l});break;case`toHaveValue`:await n.toHaveValue(o,{timeout:l});break;case`toHaveAttribute`:if(!s)return this.error(`attributeName is required for toHaveAttribute assertion`);await n.toHaveAttribute(s,o,{timeout:l});break;case`toHaveClass`:await n.toHaveClass(o,{timeout:l});break;case`toHaveCount`:await n.toHaveCount(o,{timeout:l});break;case`toHaveCSS`:if(!s)return this.error(`attributeName (CSS property name) is required for toHaveCSS assertion`);await n.toHaveCSS(s,o,{timeout:l});break;case`toHaveId`:await n.toHaveId(o,{timeout:l});break;default:return this.error(`Unsupported assertion type: ${a}`)}}return this.successJson({success:!0,assertion:c?`not.${a}`:a,selector:i??null,passed:!0})}catch(e){let t=e instanceof Error?e.message:`Assertion failed`;return this.successJson({success:!0,assertion:c?`not.${a}`:a,selector:i??null,passed:!1,failureMessage:t})}})}};Vi=zi=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ExtensionPageProxy)),z(`design:paramtypes`,[Object,Object,Object,Object])],Vi);var Hi;const Ui=W.extend({pageId:I.string().describe(`Page ID to operate on`),value:I.string().describe(`The text value to fill into the input`),force:I.boolean().optional().default(!1).describe(`Bypass actionability checks`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let Wi=class extends ${static{Hi=this}static TOOL_NAME=`browser_fill`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return Ui}getDefinition(){return{name:Hi.TOOL_NAME,description:`Fills an input field with text, clearing any existing content first. Use for form inputs, textareas, and contenteditable elements.`,inputSchema:I.toJSONSchema(Ui,{reused:`inline`})}}async execute(e){return this.executeWithMode(Hi.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=this.buildSelector(e);try{return await(await this.elementLocator.locate(t,n)).fill(e.value,{force:e.force,timeout:e.timeout}),this.success(`Filled element with: "${e.value}"`)}catch(r){let i=r instanceof Error?r.message:String(r);if(!i.includes(`Element not found`))throw r;let a=await ci(t,n,5);return this.error(JSON.stringify({error:i,pageId:e.pageId,selector:n,...a},null,2))}})}buildSelector(e){return{uid:e.uid,role:e.role,name:e.name,label:e.label,placeholder:e.placeholder,testId:e.testId,text:e.text,exact:e.exact,selector:e.selector,xpath:e.xpath,frame:e.frame}}};Wi=Hi=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],Wi);var Gi;const Ki=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),requestId:I.string().describe(`The ID of the network request to retrieve (e.g., "req-1")`)});let qi=class extends ${static{Gi=this}static TOOL_NAME=`browser_get_network_request`;constructor(e,t,n,r){super(e,t,n),this.pageMonitor=r}getInputSchema(){return Ki}getDefinition(){return{name:Gi.TOOL_NAME,description:`Retrieves detailed information about a specific network request by ID, including headers, body, and response data.`,inputSchema:I.toJSONSchema(Ki,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId);return this.executeWithMode(Gi.TOOL_NAME,t.id,e,async()=>{let n=this.pageMonitor.getNetworkRequest(t.id,e.requestId);return n?this.successJson({id:n.id,url:n.url,method:n.method,resourceType:n.resourceType,headers:n.headers,postData:n.postData,timestamp:n.timestamp.toISOString(),response:n.response?{status:n.response.status,statusText:n.response.statusText,headers:n.response.headers,timing:n.response.timing}:null}):this.error(`Network request "${e.requestId}" not found for page "${t.id}"`)})})}};qi=Gi=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.PageMonitorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],qi);var Ji;const Yi=I.object({pageId:I.string().describe(`Page ID to operate on`),waitUntil:I.enum([`load`,`domcontentloaded`,`networkidle`,`commit`]).optional().default(`load`).describe(`Wait until condition`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let Xi=class extends ${static{Ji=this}static TOOL_NAME=`browser_go_back`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Yi}getDefinition(){return{name:Ji.TOOL_NAME,description:`Navigates the browser back in history.`,inputSchema:I.toJSONSchema(Yi,{reused:`inline`})}}async execute(e){return this.executeWithMode(Ji.TOOL_NAME,e.pageId,e,async()=>{let t=await this.resolvePage(e.pageId).goBack({waitUntil:e.waitUntil,timeout:e.timeout});if(t===null)return this.success(`No previous page in history`);let n=t.status();return this.success(`Navigated back (status: ${n})`)})}};Xi=Ji=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Xi);var Zi;const Qi=I.object({pageId:I.string().describe(`Page ID to operate on`),waitUntil:I.enum([`load`,`domcontentloaded`,`networkidle`,`commit`]).optional().default(`load`).describe(`Wait until condition`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let $i=class extends ${static{Zi=this}static TOOL_NAME=`browser_go_forward`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Qi}getDefinition(){return{name:Zi.TOOL_NAME,description:`Navigates the browser forward in history.`,inputSchema:I.toJSONSchema(Qi,{reused:`inline`})}}async execute(e){return this.executeWithMode(Zi.TOOL_NAME,e.pageId,e,async()=>{let t=await this.resolvePage(e.pageId).goForward({waitUntil:e.waitUntil,timeout:e.timeout});if(t===null)return this.success(`No next page in history`);let n=t.status();return this.success(`Navigated forward (status: ${n})`)})}};$i=Zi=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],$i);var ea;const ta=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),accept:I.boolean().describe(`Whether to accept (true) or dismiss (false) the dialog`),promptText:I.string().optional().describe(`Text to enter for prompt dialogs (only used when accept is true)`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds to wait for dialog (default: ${H})`)});let na=class extends ${static{ea=this}static TOOL_NAME=`browser_handle_dialog`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return ta}getDefinition(){return{name:ea.TOOL_NAME,description:`Handles JavaScript dialogs (alert, confirm, prompt) by accepting or dismissing them. Set up a listener before triggering the action that opens the dialog.`,inputSchema:I.toJSONSchema(ta,{reused:`inline`})}}async execute(e){return this.safeExecute(()=>this.executeWithMode(ea.TOOL_NAME,e.pageId,{accept:e.accept,promptText:e.promptText,timeout:e.timeout},async()=>{let t=this.resolvePage(e.pageId),n=e.timeout??18e4,r=setTimeout(()=>{t.off(`dialog`,i)},n),i=async n=>{clearTimeout(r);try{e.accept?await n.accept(e.promptText):await n.dismiss()}finally{t.off(`dialog`,i)}};return t.once(`dialog`,i),this.successJson({armed:!0,action:e.accept?`accept`:`dismiss`,timeout:n,promptText:e.promptText??null})}))}};na=ea=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],na);var ra;const ia=W.extend({pageId:I.string().describe(`Page ID to operate on`),position:I.object({x:I.number(),y:I.number()}).optional().describe(`Position offset within element`),modifiers:I.array(I.enum([`Alt`,`Control`,`Meta`,`Shift`])).optional().describe(`Modifier keys to hold`),force:I.boolean().optional().default(!1).describe(`Bypass actionability checks`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let aa=class extends ${static{ra=this}static TOOL_NAME=`browser_hover`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return ia}getDefinition(){return{name:ra.TOOL_NAME,description:`Hovers over an element to trigger hover states, tooltips, or dropdown menus.`,inputSchema:I.toJSONSchema(ia,{reused:`inline`})}}async execute(e){return this.executeWithMode(ra.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId);return await(await this.elementLocator.locate(t,e)).hover({position:e.position,modifiers:e.modifiers,force:e.force,timeout:e.timeout}),this.success(`Hovered over element successfully`)})}};aa=ra=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],aa);var oa;const sa=W.extend({pageId:I.string().describe(`Page ID to operate on`)});let ca=class extends ${static{oa=this}static TOOL_NAME=`browser_inspect_element`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return sa}getDefinition(){return{name:oa.TOOL_NAME,description:`Inspects a matching element and returns structured metadata such as role, accessible name, label, visibility, and bounds. Use this for selector debugging and agent recovery.`,inputSchema:I.toJSONSchema(sa,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>this.executeWithMode(oa.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=this.buildSelector(e),r=await this.elementLocator.locateAll(t,n),i=await r.count();if(i===0)return this.successJson({pageId:e.pageId,selector:n,matched:!1,matchCount:0,element:null});let a=r.first(),[o,s,c,l]=await Promise.all([a.isVisible().catch(()=>!1),a.isEnabled().catch(()=>!1),a.boundingBox().catch(()=>null),a.evaluate(e=>{let t=e=>(e??``).replace(/\s+/g,` `).trim(),n=e=>{if(e instanceof HTMLInputElement||e instanceof HTMLTextAreaElement||e instanceof HTMLSelectElement){if(typeof e.labels?.length==`number`&&e.labels.length>0)return t(e.labels[0]?.textContent);if(e.id){let n=document.querySelector(`label[for="${CSS.escape(e.id)}"]`);if(n?.textContent)return t(n.textContent)}}let n=e.closest(`label`);return n?.textContent?t(n.textContent):null},r=e=>{let r=e.getAttribute(`aria-label`);if(r)return t(r);let i=e.getAttribute(`aria-labelledby`);if(i){let e=i.split(/\s+/).map(e=>t(document.getElementById(e)?.textContent)).filter(Boolean);if(e.length>0)return e.join(` `)}return n(e)||(e instanceof HTMLInputElement&&e.placeholder?t(e.placeholder):e instanceof HTMLImageElement&&e.alt?t(e.alt):t(e.textContent))},i=e=>{let t=e.getAttribute(`role`);if(t)return t;if(e instanceof HTMLInputElement)return{button:`button`,checkbox:`checkbox`,email:`textbox`,number:`spinbutton`,password:`textbox`,radio:`radio`,range:`slider`,search:`searchbox`,submit:`button`,tel:`textbox`,text:`textbox`,url:`textbox`}[(e.type||`text`).toLowerCase()]||`textbox`;let n=e.tagName.toLowerCase();return{a:e.hasAttribute(`href`)?`link`:`none`,button:`button`,img:`img`,option:`option`,select:`combobox`,textarea:`textbox`}[n]||`none`},a=e instanceof HTMLInputElement||e instanceof HTMLTextAreaElement||e instanceof HTMLSelectElement?e.value:null;return{tagName:e.tagName.toLowerCase(),id:e.id||``,className:e.getAttribute(`class`)||``,role:i(e),accessibleName:r(e),label:n(e),placeholder:e.getAttribute(`placeholder`),testId:e.getAttribute(`data-testid`)||e.getAttribute(`data-test-id`),textContent:t(e.textContent).slice(0,500),value:a}})]),u={...l,visible:o,enabled:s,rect:c?{x:c.x,y:c.y,width:c.width,height:c.height}:null};return this.successJson({pageId:e.pageId,selector:n,matched:!0,matchCount:i,element:u})}))}buildSelector(e){return{uid:e.uid,role:e.role,name:e.name,label:e.label,placeholder:e.placeholder,testId:e.testId,text:e.text,exact:e.exact,selector:e.selector,xpath:e.xpath,frame:e.frame}}};ca=oa=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],ca);var la;const ua=[`pnpm-workspace.yaml`,`nx.json`,`.git`],da=`about:blank`,fa=process.env.BROWSE_TOOL_DEBUG_EXTENSION_RECORDING===`1`,pa=`[LaunchBrowserTool]`,ma=new Set([`docker`,`docker-vm`,`docker-chromium`,`docker-chrome-testing`]);function ha(e=process.cwd()){let t=x.resolve(e);for(;;){for(let e of ua)if(k(x.join(t,e)))return t;let e=x.dirname(t);if(e===t)return process.cwd();t=e}}const ga=Le(_e);async function _a(e){try{let{stdout:t}=await ga(`git`,[`rev-parse`,`--git-common-dir`],{cwd:e}),n=t.trim(),r=x.isAbsolute(n)?n:x.join(e,n),i=x.dirname(r);return i===e?null:i}catch{return null}}const va=I.object({server:I.string().describe(`Proxy server URL (e.g., "http://proxy.example.com:8080")`),username:I.string().optional().describe(`Proxy username for authentication`),password:I.string().optional().describe(`Proxy password for authentication`),bypass:I.string().optional().describe(`Comma-separated list of hosts to bypass the proxy (e.g., "localhost,127.0.0.1")`)}),ya=I.object({mode:I.enum([`playwright`,`extension`,`vm`,`stealth`]).optional().default(`playwright`).describe(`Execution mode. "playwright" uses Playwright APIs. "extension" uses Playwright with Chrome extension. "vm" launches in an isolated runtime (for example via Docker). "stealth" spawns Chrome directly without automation flags.`),browserType:I.enum([`chromium`,`firefox`,`webkit`]).optional().default(`chromium`).describe(`Browser type to use (only applies to playwright mode)`),headless:I.boolean().optional().default(!0).describe(`Run browser in headless mode (only applies to playwright mode)`),url:I.string().optional().describe(`Optional URL to navigate to after launch`),profileName:I.string().optional().describe(`Optional profile name to use for browser settings and persistent storage`),vmCommand:I.string().optional().describe(`Executable/alias for vm mode launch (e.g., "docker" for isolated container runtime)`),vmCommandArgs:I.array(I.string()).optional().describe(`Browser arguments for vm command runtime`),vmDockerBrowser:I.enum([`cloakbrowser`,`chrome-for-testing`]).optional().describe(`Docker browser runtime for vm mode. "cloakbrowser" (default) uses stealth CloakBrowser with C++ anti-detection patches. "chrome-for-testing" uses Google Chrome for Testing.`),vmBrowserId:I.string().optional().describe(`Optional browserId override for vm mode`),startupDelayMs:I.number().optional().describe(`Optional startup delay in milliseconds before returning in vm mode`),vmServerUrl:I.string().optional().describe(`Explicit MCP server URL reachable from inside VM (e.g., "http://host.docker.internal:3200"). Overrides registry discovery.`),vmExtensionPath:I.string().optional().describe(`Explicit extension path that exists inside VM filesystem. Useful when VM does not share host workspace paths.`),vmEnableVnc:I.boolean().optional().default(!1).describe(`Expose VNC/noVNC ports when using Docker VM runtime`),vmVncPort:I.number().min(1).max(65535).optional().describe(`Host port for VNC (container port 5900) when vmEnableVnc is true`),vmNoVncPort:I.number().min(1).max(65535).optional().describe(`Host port for noVNC web UI (container port 7900) when vmEnableVnc is true`),vmVncPassword:I.string().optional().describe(`Optional VNC password for Docker VM runtime`),baseURL:I.string().optional().describe(`Base URL for relative navigation (e.g., page.goto("/") will use this as the base)`),videoDir:I.string().optional().describe(`Directory to save video recordings. If provided, video will be recorded for all pages. Videos are saved when the page or context is closed.`),proxy:I.union([va,I.literal(!0)]).optional().describe(`Proxy configuration for routing browser traffic. Pass an object for explicit proxy settings, or true to load session.yaml from the proxy config directory.`)});let ba=class{static{la=this}static TOOL_NAME=`browser_launch`;constructor(e,t,n=new X){this.browserService=e,this.stealthLauncher=t,this.proxyConfigService=n}getInputSchema(){return ya}getDefinition(){return{name:la.TOOL_NAME,description:`Launches a new browser instance. Returns browserId and pageId for subsequent browser operations. Use mode "extension" for bot-detection-free automation via Chrome extension, or mode "vm" for isolated runtime launch.`,inputSchema:I.toJSONSchema(ya,{reused:`inline`})}}async execute(e){let t=e.mode??`playwright`;try{let n={...e,proxy:this.proxyConfigService.resolve({proxy:e.proxy,profileName:e.profileName})};return t===`stealth`?await this.launchStealthMode(n):t===`vm`?await this.launchVmMode(n):t===`extension`?await this.launchExtensionMode(n):await this.launchPlaywrightMode(n)}catch(e){return{content:[{type:`text`,text:e instanceof Error?e.message:String(e)}],isError:!0}}}async launchPlaywrightMode(e){let t=await this.browserService.launch({browserType:e.browserType,headless:e.headless,profileName:e.profileName,baseURL:e.baseURL,recordVideo:e.videoDir?{dir:e.videoDir}:void 0,proxy:e.proxy});return e.url&&await t.page.goto(e.url),{content:[{type:`text`,text:JSON.stringify({browserId:t.browserInstance.id,pageId:t.pageId,mode:`playwright`,url:e.url??t.page.url()},null,2)}]}}async launchStealthMode(e){let t=await this.stealthLauncher.launch({url:e.url,profileName:e.profileName,proxy:e.proxy});return{content:[{type:`text`,text:JSON.stringify({browserId:t.id,pageId:`page-${t.id}`,mode:`stealth`,url:e.url??`about:blank`,pid:t.pid},null,2)}]}}async launchExtensionMode(e){fa&&console.warn(`[ExtensionRecordingDebug] launch tool mode=extension videoDir=${e.videoDir??``} url=${e.url??``}`);let t=e.profileName?`browser-${e.profileName}`:`browser-extension-${L()}`,n=e.url??da,r=await this.resolveExtensionBootstrapServerUrl(),i=this.buildVmBootstrapUrl(r,t,n),a=await this.browserService.launchWithExtension({url:i,baseURL:e.baseURL,profileName:e.profileName,recordVideo:e.videoDir?{dir:e.videoDir}:void 0,proxy:e.proxy,mode:`extension`,browserId:t});return{content:[{type:`text`,text:JSON.stringify({browserId:a.browserInstance.id,pageId:a.pageId,mode:`extension`,url:n,pid:a.process.pid},null,2)}]}}async launchVmMode(e){let t=e.vmBrowserId??`browser-vm-${Date.now()}-${Math.random().toString(36).slice(2,8)}`,n=e.url??da,r=e.vmCommand??`docker`,i=await this.resolveVmBootstrapServerUrl(e.vmServerUrl,r),a=this.buildVmBootstrapUrl(i,t,n),o=this.buildVmDockerRuntimeOptions(e),s=await this.browserService.launchWithExtension({url:a,baseURL:e.baseURL,profileName:e.profileName,recordVideo:e.videoDir?{dir:e.videoDir}:void 0,proxy:e.proxy,extensionPath:e.vmExtensionPath??process.env.PLAYWRIGHT_VM_EXTENSION_PATH,mode:`vm`,vmDockerBrowser:e.vmDockerBrowser,command:r,commandArgs:e.vmCommandArgs,dockerRunArgs:o.runArgs,dockerEnv:o.env,dockerEnableVnc:o.enableVnc,browserId:t,startupDelayMs:e.startupDelayMs}),c=o.vnc?{vncUrl:o.vnc.vncUrl,noVncUrl:o.vnc.noVncUrl}:void 0;return{content:[{type:`text`,text:JSON.stringify({browserId:s.browserInstance.id,pageId:s.pageId,mode:`vm`,url:n,pid:s.process.pid,...c?{vm:c}:{}},null,2)}]}}buildVmBootstrapUrl(e,t,n){let r=new URL(e);return r.searchParams.set(`playwright_mcp_connect`,`1`),r.searchParams.set(`playwright_mcp_server`,e),r.searchParams.set(`playwright_mcp_browser_id`,t),r.searchParams.set(`playwright_mcp_navigate`,n),r.toString()}async resolveVmBootstrapServerUrl(e,t){let n=e??process.env.PLAYWRIGHT_VM_SERVER_URL;if(n)return this.normalizeVmServerUrl(n);let r=K(),i=await this.getRegisteredPort(r);return this.isDockerVmCommand(t??`docker`)?q(`host.docker.internal`,i.port):q(i.host,i.port)}async resolveExtensionBootstrapServerUrl(){let e=K(),t=await this.getRegisteredPort(e);return q(t.host,t.port)}normalizeVmServerUrl(e){let t;try{t=new URL(e)}catch{throw Error(`Invalid vmServerUrl: "${e}"`)}if(t.protocol!==`http:`&&t.protocol!==`https:`)throw Error(`vmServerUrl must use http or https: "${e}"`);return t.origin}isDockerVmCommand(e){return ma.has(e.trim().toLowerCase())}buildVmDockerRuntimeOptions(e){let t=e.vmEnableVnc||e.vmVncPort!==void 0||e.vmNoVncPort!==void 0,n=[],r={},i;if(t){let t=e.vmVncPort??5901,r=e.vmNoVncPort??7901;n.push(`-p`,`${t}:5900`,`-p`,`${r}:7900`),i={vncUrl:`vnc://localhost:${t}`,noVncUrl:`http://localhost:${r}/`}}return e.vmVncPassword&&e.vmVncPassword.trim().length>0&&(r.SE_VNC_PASSWORD=e.vmVncPassword.trim()),{runArgs:n.length>0?n:void 0,env:Object.keys(r).length>0?r:void 0,enableVnc:t,vnc:i}}getFallbackPort(){try{return Fn()}catch{return Nn}}isValidPort(e){return typeof e==`number`&&Number.isInteger(e)&&e>0&&e<=65535}async getRegisteredPort(e){let t={host:e,port:this.getFallbackPort()},n=ha(process.cwd()),r=process.env.NODE_ENV||`development`,i=new d(process.env.PORT_REGISTRY_PATH),a={serviceName:`browse-tool-http`,serviceType:`tool`,environment:r},o=async t=>{try{let n=await i.getPort({...a,repositoryPath:t});if(n.success&&n.record&&this.isValidPort(n.record.port))return{host:n.record.host||e,port:n.record.port}}catch(e){let n=e instanceof Error?e.message:String(e);console.warn(`${pa} Registry lookup failed (path=${t}): ${n}`)}return null},s=await o(n);if(s)return s;let c=await _a(n);if(c){let e=await o(c);if(e)return e}try{let t=(await i.listAllocations(a)).filter(e=>this.isValidPort(e.port)).sort((e,t)=>new Date(t.updatedAt).getTime()-new Date(e.updatedAt).getTime())[0];if(t)return{host:t.host||e,port:t.port}}catch(e){let t=e instanceof Error?e.message:String(e);console.warn(`${pa} Global registry lookup failed: ${t}`)}return t}};ba=la=V([v(),B(0,_(R.BrowserService)),B(1,_(R.StealthLauncher)),B(2,_(R.ProxyConfigService)),z(`design:paramtypes`,[Object,Object,Object])],ba);var xa;const Sa=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),type:I.enum([`log`,`info`,`warning`,`error`,`debug`,`trace`,`dir`,`dirxml`,`table`,`count`,`assert`]).optional().describe(`Filter by message type (log, info, warning, error, debug, trace, etc.)`),limit:I.number().min(1).optional().describe(`Maximum number of messages to return`)});let Ca=class extends ${static{xa=this}static TOOL_NAME=`browser_list_console_messages`;constructor(e,t,n,r){super(e,t,n),this.pageMonitor=r}getInputSchema(){return Sa}getDefinition(){return{name:xa.TOOL_NAME,description:`Lists all console messages captured during page automation. Requires monitoring to be started first.`,inputSchema:I.toJSONSchema(Sa,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId);return this.executeWithMode(xa.TOOL_NAME,t.id,e,async()=>{let n=this.pageMonitor.getConsoleMessages(t.id,e.type);return e.limit&&(n=n.slice(0,e.limit)),this.successJson({pageId:t.id,messageCount:n.length,messages:n.map(e=>({id:e.id,type:e.type,text:e.text,location:e.location,timestamp:e.timestamp.toISOString()}))})})})}};Ca=xa=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.PageMonitorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],Ca);var wa;const Ta=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),resourceType:I.string().optional().describe(`Filter by resource type (document, xhr, fetch, script, stylesheet, image, etc.)`),urlPattern:I.string().optional().describe(`Filter by URL pattern (substring match)`),method:I.string().optional().describe(`Filter by HTTP method (GET, POST, etc.)`),limit:I.number().min(1).optional().describe(`Maximum number of requests to return`)});let Ea=class extends ${static{wa=this}static TOOL_NAME=`browser_list_network_requests`;constructor(e,t,n,r){super(e,t,n),this.pageMonitor=r}getInputSchema(){return Ta}getDefinition(){return{name:wa.TOOL_NAME,description:`Lists all network requests captured during page automation. Requires monitoring to be started first.`,inputSchema:I.toJSONSchema(Ta,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId);return this.executeWithMode(wa.TOOL_NAME,t.id,e,async()=>{let n=this.pageMonitor.getNetworkRequests(t.id);if(e.resourceType&&(n=n.filter(t=>t.resourceType===e.resourceType)),e.urlPattern){let t=e.urlPattern;n=n.filter(e=>e.url.includes(t))}return e.method&&(n=n.filter(t=>t.method===e.method)),e.limit&&(n=n.slice(0,e.limit)),this.successJson({pageId:t.id,requestCount:n.length,requests:n.map(e=>({id:e.id,url:e.url,method:e.method,resourceType:e.resourceType,status:e.response?.status,statusText:e.response?.statusText,timing:e.response?.timing,timestamp:e.timestamp.toISOString()}))})})})}};Ea=wa=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.PageMonitorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],Ea);var Da;const Oa=I.object({browserId:I.string().describe(`Browser ID to list pages for`)});let ka=class extends ${static{Da=this}static TOOL_NAME=`browser_list_pages`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Oa}getDefinition(){return{name:Da.TOOL_NAME,description:`Lists all pages (tabs) in a browser. Returns page IDs, URLs, titles, and indicates which page is currently active.`,inputSchema:I.toJSONSchema(Oa,{reused:`inline`})}}async execute(e){return this.toolExecutor.executeForBrowser(Da.TOOL_NAME,e.browserId,e,async()=>{let t=this.browserService.getBrowser(e.browserId);if(!t)return this.error(`Browser "${e.browserId}" not found`);let n=this.pageRegistry.findByBrowser(e.browserId),r=await Promise.all(n.map(async e=>{await this.pageRegistry.updateMetadata(e.id);let n=this.pageRegistry.get(e.id);return{pageId:e.id,url:n?.url??e.url,title:n?.title??e.title,isActive:e.id===t.currentPageId,browserId:e.browserId,profileName:e.profileName,createdAt:e.createdAt.toISOString()}}));return this.successJson({browserId:e.browserId,currentPageId:t.currentPageId,pageCount:r.length,pages:r})})}};ka=Da=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],ka);var Aa;const ja=I.object({});let Ma=class{static{Aa=this}static TOOL_NAME=`browser_list_profiles`;constructor(e){this.profileService=e}getInputSchema(){return ja}getDefinition(){return{name:Aa.TOOL_NAME,description:`Lists all available browser profiles with their settings, browser type, viewport, and timestamps.`,inputSchema:I.toJSONSchema(ja,{reused:`inline`})}}async execute(){try{let e=await this.profileService.list();return{content:[{type:`text`,text:JSON.stringify({profileCount:e.length,profiles:e.map(e=>({name:e.name,browserType:e.browserType,viewport:e.viewport??null,userAgent:e.userAgent??null,locale:e.locale??null,timezone:e.timezone??null,colorScheme:e.colorScheme??null,createdAt:e.createdAt,updatedAt:e.updatedAt}))},null,2)}]}}catch(e){return{content:[{type:`text`,text:`Error: ${e instanceof Error?e.message:`Unknown error`}`}],isError:!0}}}};Ma=Aa=V([v(),B(0,_(R.ProfileService)),z(`design:paramtypes`,[Object])],Ma);var Na;const Pa=I.object({});let Fa=class{static{Na=this}static TOOL_NAME=`browser_list_snippets`;constructor(e){this.snippetService=e}getInputSchema(){return Pa}getDefinition(){return{name:Na.TOOL_NAME,description:`Lists saved code snippets by name, description, and snippetPath for reuse with browser_run_code.`,inputSchema:I.toJSONSchema(Pa,{reused:`inline`})}}async execute(){try{let e=await this.snippetService.listSnippets();return{content:[{type:`text`,text:JSON.stringify({snippetCount:e.length,snippets:e},null,2)}]}}catch(e){return{content:[{type:`text`,text:`Error: ${e instanceof Error?e.message:`Unknown error`}`}],isError:!0}}}};Fa=Na=V([v(),B(0,_(R.CodeSnippetService)),z(`design:paramtypes`,[Object])],Fa);var Ia;const La=I.object({pageId:I.string().describe(`Page ID to operate on`),url:I.string().describe(`URL to navigate to`),waitUntil:I.enum([`load`,`domcontentloaded`,`networkidle`,`commit`]).optional().default(`load`).describe(`Wait until condition`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`),referer:I.string().optional().describe(`Referer header value`)});let Ra=class extends ${static{Ia=this}static TOOL_NAME=`browser_navigate`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return La}getDefinition(){return{name:Ia.TOOL_NAME,description:`Navigates the browser to a specified URL.`,inputSchema:I.toJSONSchema(La,{reused:`inline`})}}async execute(e){return this.executeWithMode(Ia.TOOL_NAME,e.pageId,e,async()=>{let t=(await this.resolvePage(e.pageId).goto(e.url,{waitUntil:e.waitUntil,timeout:e.timeout,referer:e.referer}))?.status()??`unknown`;return this.success(`Navigated to ${e.url} (status: ${t})`)})}};Ra=Ia=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Ra);var za;const Ba=I.object({browserId:I.string().optional().describe(`Browser ID to create the page in (optional - uses default browser if not specified)`),url:I.string().optional().describe(`Optional URL to navigate to after creation`),setAsCurrent:I.boolean().optional().default(!0).describe(`Whether to set as current page for the browser (default: true)`)});let Va=class extends ${static{za=this}static TOOL_NAME=`browser_new_page`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Ba}getDefinition(){return{name:za.TOOL_NAME,description:`Creates a new page (tab) in a browser instance. Uses the default browser if browserId is not specified, launching one if needed.`,inputSchema:I.toJSONSchema(Ba,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=e.browserId?this.browserService.getBrowser(e.browserId):await this.browserService.getOrCreateDefaultBrowser();if(!t)return this.error(`Browser "${e.browserId}" not found`);let n=t.id,r=e.setAsCurrent!==!1;if(t.mode===`extension`||t.mode===`vm`){let i=this.pageRegistry.registerExtensionPage(n,void 0,e.url,!1),a=await this.toolExecutor.executeForBrowser(za.TOOL_NAME,n,{...e,pageId:i},async()=>this.error(`Browser "${n}" is not in extension mode`));if(a.isError)return this.pageRegistry.remove(i),a;t.pageIds.add(i),(r||!t.currentPageId)&&(t.currentPageId=i);let o=a.content[0]?.type===`text`?a.content[0].text:void 0,s={};if(typeof o==`string`)try{s=JSON.parse(o)}catch{s={}}let c=this.pageRegistry.get(i);return c&&(c.url=s.url??c.url,c.title=s.title??c.title,c.extensionTabId=s.tabId??c.extensionTabId),this.browserService.recordBrowserActivity(n,i),this.successJson({pageId:i,url:c?.url??e.url??`about:blank`,title:c?.title??``,browserId:n,isActive:t.currentPageId===i})}let{pageId:i,page:a}=await this.browserService.newPage(n);if(e.url)try{await a.goto(e.url),await this.pageRegistry.updateMetadata(i)}catch(e){throw await a.close(),e}r&&this.browserService.setCurrentPage(n,i);let o=this.pageRegistry.get(i);return this.successJson({pageId:i,url:o?.url??a.url(),title:o?.title??``,browserId:n,isActive:r})})}};Va=za=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Va);var Ha;const Ua=I.object({pageId:I.string().describe(`Page ID to operate on`),format:I.enum([`Letter`,`Legal`,`Tabloid`,`Ledger`,`A0`,`A1`,`A2`,`A3`,`A4`,`A5`,`A6`]).optional().default(`Letter`).describe(`Paper format`),printBackground:I.boolean().optional().default(!1).describe(`Print background graphics`),landscape:I.boolean().optional().default(!1).describe(`Landscape orientation`),scale:I.number().min(.1).max(2).optional().default(1).describe(`Page scale (0.1 to 2)`),marginTop:I.string().optional().describe(`Top margin (e.g., "1cm", "0.5in")`),marginBottom:I.string().optional().describe(`Bottom margin`),marginLeft:I.string().optional().describe(`Left margin`),marginRight:I.string().optional().describe(`Right margin`),filename:I.string().optional().describe(`Custom filename without extension`)});let Wa=class extends ${static{Ha=this}static TOOL_NAME=`browser_pdf`;constructor(e,t,n,r){super(e,t,n),this.extensionTaskQueue=r}getInputSchema(){return Ua}getDefinition(){return{name:Ha.TOOL_NAME,description:`Generates a PDF of the page with configurable format options. Saves to temp directory and returns the file path.`,inputSchema:I.toJSONSchema(Ua,{reused:`inline`})}}async execute(e){let t=this.getPage(e.pageId);return t?t.mode===`extension`?this.safeExecute(async()=>{let n=e.filename??`page-${L()}`,r=T(P(),`${n}.pdf`),i=await this.extensionTaskQueue.queueTask(Ha.TOOL_NAME,{pageId:e.pageId,format:e.format,printBackground:e.printBackground,landscape:e.landscape,scale:e.scale,marginTop:e.marginTop,marginBottom:e.marginBottom,marginLeft:e.marginLeft,marginRight:e.marginRight,filename:n},void 0,t.browserId),a=JSON.parse(i.result?.content?.[0]?.text??`{}`);if(!a.pdfBase64)throw Error(`Extension mode PDF generation returned no data`);return await M(r,Buffer.from(a.pdfBase64,`base64`)),this.success(`PDF saved to: ${r}`)}):this.executeWithMode(Ha.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=e.filename??`page-${L()}`,r=T(P(),`${n}.pdf`);return await t.pdf({path:r,format:e.format??`Letter`,printBackground:e.printBackground??!1,landscape:e.landscape??!1,scale:e.scale??1,margin:{top:e.marginTop,bottom:e.marginBottom,left:e.marginLeft,right:e.marginRight}}),this.success(`PDF saved to: ${r}`)}):this.error(`Page "${e.pageId}" not found`)}};Wa=Ha=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ExtensionTaskQueue)),z(`design:paramtypes`,[Object,Object,Object,Object])],Wa);var Ga;const Ka=I.object({pageId:I.string().describe(`Page ID to operate on`),key:I.string().describe(`Key or key combination (e.g., "Enter", "Tab", "Control+a", "Meta+Shift+t", "ArrowDown")`),element:W.optional().describe(`Optional element to focus before pressing key`),delay:I.number().optional().describe(`Delay between keydown and keyup in ms`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let qa=class extends ${static{Ga=this}static TOOL_NAME=`browser_press_key`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return Ka}getDefinition(){return{name:Ga.TOOL_NAME,description:`Presses a keyboard key or key combination. Supports modifier keys like Control, Shift, Alt, Meta.`,inputSchema:I.toJSONSchema(Ka,{reused:`inline`})}}async execute(e){return this.executeWithMode(Ga.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId);return e.element?await(await this.elementLocator.locate(t,e.element)).press(e.key,{delay:e.delay,timeout:e.timeout}):await t.keyboard.press(e.key,{delay:e.delay}),this.success(`Pressed key: "${e.key}"`)})}};qa=Ga=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],qa);var Ja;const Ya=I.object({pageId:I.string().describe(`Page ID to operate on`),waitUntil:I.enum([`load`,`domcontentloaded`,`networkidle`,`commit`]).optional().default(`load`).describe(`Wait until condition`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let Xa=class extends ${static{Ja=this}static TOOL_NAME=`browser_reload`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Ya}getDefinition(){return{name:Ja.TOOL_NAME,description:`Reloads the current page.`,inputSchema:I.toJSONSchema(Ya,{reused:`inline`})}}async execute(e){return this.executeWithMode(Ja.TOOL_NAME,e.pageId,e,async()=>{let t=(await this.resolvePage(e.pageId).reload({waitUntil:e.waitUntil,timeout:e.timeout}))?.status()??`unknown`;return this.success(`Page reloaded (status: ${t})`)})}};Xa=Ja=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Xa);var Za;const Qa=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),width:I.number().min(1).describe(`Viewport width in pixels`),height:I.number().min(1).describe(`Viewport height in pixels`),deviceScaleFactor:I.number().min(1).max(3).optional().describe(`Device scale factor (default: 1)`),isMobile:I.boolean().optional().describe(`Whether the meta viewport tag is taken into account`),hasTouch:I.boolean().optional().describe(`Whether the viewport supports touch events`)});let $a=class extends ${static{Za=this}static TOOL_NAME=`browser_resize_page`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Qa}getDefinition(){return{name:Za.TOOL_NAME,description:`Resizes the viewport of a page to specified width and height dimensions.`,inputSchema:I.toJSONSchema(Qa,{reused:`inline`})}}async execute(e){return this.executeWithMode(Za.TOOL_NAME,e.pageId,{width:e.width,height:e.height},async()=>{let t=this.resolvePage(e.pageId);await t.setViewportSize({width:e.width,height:e.height});let n=t.viewportSize();return this.successJson({success:!0,viewport:{width:n?.width??e.width,height:n?.height??e.height}})})}};$a=Za=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],$a);var eo;const to=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),code:I.string().optional().describe(`JavaScript code to execute. Receives { page, context, browser } as arguments. Example: "await page.goto('https://example.com'); return await page.title();"`),snippetPath:I.string().optional().describe(`Relative snippet path inside the configured snippets directory. When provided, executes the saved snippet instead of inline code.`),saveAs:I.object({name:I.string().describe(`Human-readable snippet name. The filename is generated from a sanitized version of this value.`),description:I.string().describe(`Short description shown by browser_list_snippets.`)}).optional().describe(`Optional snippet metadata used to save the inline code into the configured snippets directory.`)});let no=class extends ${static{eo=this}static TOOL_NAME=`browser_run_code`;constructor(e,t,n,r,i){super(e,t,n),this.extensionPageProxy=r,this.snippetService=i}getInputSchema(){return to}getDefinition(){return{name:eo.TOOL_NAME,description:`Runs a Playwright code snippet. The code should be an async function body that receives { page, context, browser } and can perform any Playwright operation. Return a value to include it in the response.`,inputSchema:I.toJSONSchema(to,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{if(!e.code&&!e.snippetPath)return this.error(`Either "code" or "snippetPath" is required`);if(e.saveAs&&!e.code)return this.error(`"saveAs" requires inline "code"`);let t=this.resolvePageEntry(e.pageId),{page:n,context:r,browser:i}=t,a=t.mode===`extension`?(this.extensionPageProxy.setTarget(t.id,t.browserId),this.extensionPageProxy):n,o=t.mode===`extension`?this.extensionPageProxy.context():r,s=t.mode===`extension`?{id:t.browserId,mode:this.browserService.getBrowser(t.browserId)?.mode??`extension`,pageId:t.id}:i,c;e.saveAs&&(c=await this.snippetService.saveSnippet({name:e.saveAs.name,description:e.saveAs.description,code:e.code}));let l=e.snippetPath?await this.snippetService.loadSnippet(e.snippetPath).then(e=>e.run({page:a,context:o,browser:s})):await Function(`page`,`context`,`browser`,`return (async () => { ${e.code} })();`)(a,o,s);return l===void 0?c?this.successJson({savedSnippet:c}):this.success(`Code executed successfully`):this.successJson(c?{result:l,savedSnippet:c}:{result:l})})}};no=eo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ExtensionPageProxy)),B(4,_(R.CodeSnippetService)),z(`design:paramtypes`,[Object,Object,Object,Object,Object])],no);var ro;const io={width:1920,height:1080},ao=I.object({specPath:I.string().describe(`Absolute path to the Playwright spec file to execute`),automationId:I.string().optional().describe(`Optional automation session ID (auto-generated if not provided)`),mode:I.enum([`playwright`,`extension`]).optional().default(`playwright`).describe(`Execution mode: playwright (standard) or extension (Chrome extension)`),browserId:I.string().optional().describe(`Browser ID to use (required for extension mode)`),pageId:I.string().optional().describe(`Page ID to use (required for extension mode)`),browserType:I.enum([`chromium`,`firefox`,`webkit`]).optional().default(`chromium`).describe(`Browser type to use (only applies to playwright mode)`),headless:I.boolean().optional().default(!0).describe(`Run browser in headless mode (only applies to playwright mode)`),baseURL:I.string().optional().describe(`Base URL for relative navigation (e.g., page.goto("/") will use this as the base)`),keepBrowserOpen:I.boolean().optional().default(!1).describe(`Keep browser open after spec execution completes for further interaction`),hooksPath:I.string().optional().describe(`Explicit absolute path to run-spec-hooks.ts file. Skips auto-discovery when provided. The source directory is injected as process.env.__HOOKS_SOURCE_DIR__ so hooks can resolve paths relative to their original location.`),env:I.record(I.string(),I.string()).optional().describe(`Environment variables to set before running the spec. Applied before hooks execution. Useful for passing config like API endpoints without relying on hooks file path resolution.`),testName:I.string().optional().describe(`Filter by exact test name`),testPattern:I.string().optional().describe(`Filter by regex pattern on full test title (e.g., "login.*success")`),onlyMarked:I.boolean().optional().default(!1).describe(`Run only tests marked with test.only()`),describeFilter:I.string().optional().describe(`Filter by describe block name pattern`),specArgs:I.record(I.string(),I.unknown()).optional().describe(`Arguments to pass to the spec (available as fixtures.specArgs)`),loadArgsFromEnv:I.boolean().optional().default(!1).describe(`Load specArgs from environment variables using spec's envPrefix export`),videoDir:I.string().optional().describe(`Absolute directory path to save video recordings. Supported in both playwright and extension modes.`)});let oo=class{static{ro=this}static TOOL_NAME=`run_spec`;constructor(e,t,n,r,i){this.automationRunner=e,this.specMetadataService=t,this.extensionSpecRunner=n,this.extensionPageProxy=r,this.specBundlerService=i}getInputSchema(){return ao}getDefinition(){return{name:ro.TOOL_NAME,description:`Runs a Playwright spec file with optional test filtering and dynamic arguments. Supports filtering by test name, pattern, or test.only markers.`,inputSchema:I.toJSONSchema(ao,{reused:`inline`})}}async execute(e){if(!w(e.specPath))return{content:[{type:`text`,text:`Error: specPath must be an absolute path (e.g., /Users/me/project/tests/my-spec.ts)`}],isError:!0};if(e.hooksPath&&!w(e.hooksPath))return{content:[{type:`text`,text:`Error: hooksPath must be an absolute path`}],isError:!0};if(e.videoDir&&!w(e.videoDir))return{content:[{type:`text`,text:`Error: videoDir must be an absolute path`}],isError:!0};try{if(e.env)for(let[t,n]of Object.entries(e.env))process.env[t]=n;let t=this.resolveHooksPath(e.specPath,e.hooksPath),n=t?await this.runHooks(t):null;return n?.baseURL&&!e.baseURL&&(e.baseURL=n.baseURL),(e.mode??`playwright`)===`extension`?await this.executeExtensionMode(e):e.testName||e.testPattern||e.onlyMarked||e.describeFilter||e.loadArgsFromEnv?await this.executeEnhanced(e,t):await this.executeBasic(e,t)}catch(e){return{content:[{type:`text`,text:`Error: ${e instanceof Error?e.message:`Unknown error`}`}],isError:!0}}}findHooksFile(e){let t=C(e);for(let e=0;e<4;e++){let e=T(t,`run-spec-hooks.ts`);if(k(e))return e;let n=C(t);if(n===t)break;t=n}return null}async runHooks(e){let t=C(D(e)),n=await this.specBundlerService.bundle(e);try{process.env.__HOOKS_SOURCE_DIR__=t;let e=await import(`${n.outputPath}?t=${Date.now()}`);return typeof e.setup==`function`?await e.setup():void 0}finally{process.env.__HOOKS_SOURCE_DIR__=void 0,await n.cleanup()}}resolveHooksPath(e,t){return t?D(t):this.findHooksFile(e)}async executeExtensionMode(e){if(!e.pageId||!e.browserId)return{content:[{type:`text`,text:`Error: pageId and browserId are required for extension mode`}],isError:!0};this.extensionPageProxy.setTarget(e.pageId,e.browserId);let t=e.specArgs??{};if(e.loadArgsFromEnv){let n=await this.specMetadataService.extractSpecMetadata(e.specPath);n.argsSchema&&n.envPrefix&&(t={...this.specMetadataService.parseArgsFromEnv(n.argsSchema,n.envPrefix),...t})}let n={};e.testName&&(n.testName=e.testName),e.testPattern&&(n.testPattern=e.testPattern),e.onlyMarked&&(n.onlyMarked=e.onlyMarked),e.describeFilter&&(n.describeFilter=e.describeFilter);let r=e.automationId??`ext-spec-${Date.now()}`,i=!!e.videoDir;i&&await this.extensionPageProxy.startRecording();let a;try{a=await this.extensionSpecRunner.executeSpec({specPath:e.specPath,sessionId:r,testFilter:Object.keys(n).length>0?n:void 0,specArgs:Object.keys(t).length>0?t:void 0})}finally{if(i&&e.videoDir)try{let t=await this.extensionPageProxy.stopRecording();if(t){ne(e.videoDir,{recursive:!0});let n=T(e.videoDir,`recording-${r}.webm`);ie(n,Buffer.from(t,`base64`)),a.videoPath=n}}catch{}}return this.formatExtensionResult(a)}async executeBasic(e,t){let n=await this.automationRunner.runSpec({specPath:e.specPath,automationId:e.automationId,browserId:e.browserId,pageId:e.pageId,keepBrowserOpen:e.keepBrowserOpen??!1,recordVideo:e.videoDir?{dir:e.videoDir,size:io}:void 0,browserOptions:{browserType:e.browserType,headless:e.headless,baseURL:e.baseURL},hooksPath:t??void 0});return this.formatResult(n,e.keepBrowserOpen??!1)}async executeEnhanced(e,t){let n=e.specArgs??{};if(e.loadArgsFromEnv){let t=await this.specMetadataService.extractSpecMetadata(e.specPath);t.argsSchema&&t.envPrefix&&(n={...this.specMetadataService.parseArgsFromEnv(t.argsSchema,t.envPrefix),...n})}let r={};e.testName&&(r.testName=e.testName),e.testPattern&&(r.testPattern=e.testPattern),e.onlyMarked&&(r.onlyMarked=e.onlyMarked),e.describeFilter&&(r.describeFilter=e.describeFilter);let i=await this.automationRunner.runSpecEnhanced({specPath:e.specPath,automationId:e.automationId,browserId:e.browserId,pageId:e.pageId,keepBrowserOpen:e.keepBrowserOpen??!1,recordVideo:e.videoDir?{dir:e.videoDir,size:io}:void 0,browserOptions:{browserType:e.browserType,headless:e.headless,baseURL:e.baseURL},testFilter:Object.keys(r).length>0?r:void 0,specArgs:Object.keys(n).length>0?n:void 0,hooksPath:t??void 0});return this.formatResult(i,e.keepBrowserOpen??!1)}formatResult(e,t){let n=e.status===`completed`?t?`Spec completed. Browser kept open for further interaction.`:`Spec completed.`:t?`Spec failed. Browser kept open for further interaction.`:`Spec failed.`,r=e.status!==`completed`||e.specResult?.success===!1;return{content:[{type:`text`,text:JSON.stringify({automationId:e.automationId,status:e.status,pageIds:e.pageIds,browserId:e.browserId,specResult:e.specResult,warnings:e.specResult?.warnings,message:n},null,2)}],...r?{isError:!0}:{}}}formatExtensionResult(e){let t=e.success?`Spec completed: ${e.passed}/${e.totalTests} tests passed`:`Spec failed: ${e.failed}/${e.totalTests} tests failed`;return{content:[{type:`text`,text:JSON.stringify({sessionId:e.sessionId,status:e.success?`completed`:`failed`,totalTests:e.totalTests,passed:e.passed,failed:e.failed,duration:e.duration,handoffOccurred:e.handoffOccurred,testResults:e.testResults,videoPath:e.videoPath,warnings:e.warnings,message:t},null,2)}],isError:!e.success}}};oo=ro=V([v(),B(0,_(R.AutomationRunner)),B(1,_(R.SpecMetadataService)),B(2,_(R.ExtensionSpecRunner)),B(3,_(R.ExtensionPageProxy)),B(4,_(R.SpecBundlerService)),z(`design:paramtypes`,[Object,Object,Object,Object,Object])],oo);var so;const co=I.object({profileName:I.string().describe(`Profile to save state into`),browserId:I.string().optional().describe(`Browser to read state from`)});let lo=class{static{so=this}static TOOL_NAME=`browser_save_profile_state`;constructor(e,t,n){this.profileService=e,this.browserService=t,this.extensionPageProxy=n}getInputSchema(){return co}getDefinition(){return{name:so.TOOL_NAME,description:`Saves the current browser state to the named profile for reuse across future launches.`,inputSchema:I.toJSONSchema(co,{reused:`inline`})}}async execute(e){try{let t=e.browserId?this.browserService.getBrowser(e.browserId):this.browserService.getBrowserByProfile(e.profileName);if(!t)throw Error(`Browser for profile "${e.profileName}" not found`);if(t.context){let n=await t.context.storageState();await this.profileService.saveStorageState(e.profileName,n)}else{let n=t.currentPageId??[...t.pageIds][0];if(!n)throw Error(`Browser "${t.id}" has no active page`);this.extensionPageProxy.setTarget(n,t.id);let r=await this.extensionPageProxy.getStorageState();await this.profileService.saveStorageState(e.profileName,r)}return{content:[{type:`text`,text:JSON.stringify({profileName:e.profileName,browserId:t.id,saved:!0},null,2)}]}}catch(e){return{content:[{type:`text`,text:e instanceof Error?e.message:String(e)}],isError:!0}}}};lo=so=V([v(),B(0,_(R.ProfileService)),B(1,_(R.BrowserService)),B(2,_(R.ExtensionPageProxy)),z(`design:paramtypes`,[Object,Object,Object])],lo);var uo;const fo=I.object({pageId:I.string().describe(`Page ID to operate on`),fullPage:I.boolean().optional().default(!1).describe(`Capture full page scrollable area`),selector:I.string().optional().describe(`CSS selector for element to screenshot`),clip:I.object({x:I.number().min(0).describe(`X coordinate of top-left corner`),y:I.number().min(0).describe(`Y coordinate of top-left corner`),width:I.number().min(1).describe(`Width of clipping region`),height:I.number().min(1).describe(`Height of clipping region`)}).optional().describe(`Clip region to capture by coordinates (cannot be used with selector)`),type:I.enum([`png`,`jpeg`]).optional().default(`png`).describe(`Image format`),quality:I.number().min(0).max(100).optional().describe(`Quality for jpeg format (0-100)`),omitBackground:I.boolean().optional().default(!1).describe(`Omit background for transparent PNG`),filename:I.string().optional().describe(`Custom filename without extension`)});let po=class extends ${static{uo=this}static TOOL_NAME=`browser_screenshot`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return fo}getDefinition(){return{name:uo.TOOL_NAME,description:`Captures a screenshot of the page or specific element. Saves to temp directory and returns the file path.`,inputSchema:I.toJSONSchema(fo,{reused:`inline`})}}async execute(e){return this.executeWithMode(uo.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId);if(e.clip&&e.selector)throw Error(`Cannot use both clip and selector options together`);let n=e.type??`png`,r=e.filename??`screenshot-${L()}`,i=T(P(),`${r}.${n}`);if(e.selector){let r=await t.$(e.selector);if(!r)throw Error(`Element not found: ${e.selector}`);await r.screenshot({path:i,type:n,omitBackground:e.omitBackground??!1,quality:n===`jpeg`?e.quality:void 0})}else await t.screenshot({path:i,type:n,fullPage:e.fullPage??!1,omitBackground:e.omitBackground??!1,quality:n===`jpeg`?e.quality:void 0,clip:e.clip});return this.success(`Screenshot saved to: ${i}`)})}};po=uo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],po);var mo;const ho=I.object({pageId:I.string().describe(`Page ID to set as current`)});let go=class extends ${static{mo=this}static TOOL_NAME=`browser_select_page`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return ho}getDefinition(){return{name:mo.TOOL_NAME,description:`Switches the active page for a browser. Updates the current page ID so subsequent operations use this page by default.`,inputSchema:I.toJSONSchema(ho,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.pageRegistry.get(e.pageId);if(!t)return this.error(`Page "${e.pageId}" not found`);let{browserId:n}=t;if(t.mode===`extension`||t.mode===`vm`){let t=await this.executeWithMode(mo.TOOL_NAME,e.pageId,e,async()=>this.error(`Page "${e.pageId}" is not in extension mode`));if(t.isError)return t}this.browserService.setCurrentPage(n,e.pageId),this.browserService.recordBrowserActivity(n,e.pageId),await this.pageRegistry.updateMetadata(e.pageId);let r=this.pageRegistry.get(e.pageId);return this.successJson({browserId:n,pageId:e.pageId,url:r?.url??t.url,title:r?.title??t.title,message:`Page "${e.pageId}" is now the active page for browser "${n}"`})})}};go=mo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],go);var _o;const vo=W.extend({pageId:I.string().describe(`Page ID to operate on`),values:I.array(I.string()).optional().describe(`Option value(s) to select`),labels:I.array(I.string()).optional().describe(`Option label(s) to select`),indexes:I.array(I.number()).optional().describe(`Option index(es) to select`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let yo=class extends ${static{_o=this}static TOOL_NAME=`browser_select`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return vo}getDefinition(){return{name:_o.TOOL_NAME,description:`Selects option(s) from a dropdown/select element. Can select by value, label, or index.`,inputSchema:I.toJSONSchema(vo,{reused:`inline`})}}async execute(e){return this.executeWithMode(_o.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=await this.elementLocator.locate(t,e),r=[];e.values?r=e.values.map(e=>({value:e})):e.labels?r=e.labels.map(e=>({label:e})):e.indexes&&(r=e.indexes.map(e=>({index:e})));let i=await n.selectOption(r,{timeout:e.timeout});return this.success(`Selected ${i.length} option(s): ${i.join(`, `)}`)})}};yo=_o=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],yo);var bo;const xo=I.object({pageId:I.string().describe(`Page ID to operate on`),root:I.string().optional().describe(`CSS selector for root element to snapshot (defaults to body)`),sizeThreshold:I.number().optional().default(1e4).describe(`Size threshold in bytes; snapshots larger than this are saved to file (default: 10000)`)});let So=class extends ${static{bo=this}static TOOL_NAME=`browser_snapshot`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return xo}getDefinition(){return{name:bo.TOOL_NAME,description:`Returns the accessibility tree snapshot of the page. Preferred for LLM processing as it provides structured, semantic information about page content.`,inputSchema:I.toJSONSchema(xo,{reused:`inline`})}}async execute(e){return this.executeWithMode(bo.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId),n=await this.captureSnapshot(t,e.root);if(!n)return this.success(`No accessibility tree available for this page`);let r=e.sizeThreshold??1e4;if(Buffer.byteLength(n,`utf-8`)<=r)return this.success(n);let i=`snapshot-${L()}.yaml`,a=T(P(),i);return await M(a,n,`utf-8`),this.success(`Snapshot saved to: ${a}`)})}async captureSnapshot(e,t){let n=t?[t]:[`body`,`html`];for(let t of n){let n=e.locator(t).first();if(await n.count()!==0)try{return await n.ariaSnapshot()}catch{}}if(!t){await e.waitForTimeout(250);for(let t of[`body`,`html`]){let n=e.locator(t).first();if(await n.count()!==0)try{return await n.ariaSnapshot()}catch{}}}throw Error(`Root element not found: ${t??`body/html`}`)}};So=bo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],So);var Co;const wo=I.object({pageId:I.string().describe(`The page ID to record.`),path:I.string().optional().describe(`Optional absolute output path for the .webm file. If omitted, browse-tool creates a temporary recording path.`)});let To=class extends ${static{Co=this}static TOOL_NAME=`browser_start_recording`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return wo}getDefinition(){return{name:Co.TOOL_NAME,description:`Starts recording an existing extension/vm browser page and saves the artifact to the provided path or a temporary .webm file.`,inputSchema:I.toJSONSchema(wo,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId);if(t.mode!==`extension`&&t.mode!==`vm`)return this.error(`Page "${e.pageId}" is in "${t.mode}" mode and cannot use extension recording`);if(e.path){if(!x.isAbsolute(e.path))return this.error(`Recording path must be an absolute path (e.g., /Users/me/project/recording.webm)`);if(x.extname(e.path).toLowerCase()!==`.webm`)return this.error(`Recording path must end with .webm`)}let n=await this.browserService.startPageRecording(t.browserId,t.id,e.path);return this.successJson({started:!0,browserId:t.browserId,pageId:t.id,outputPath:n.outputPath})})}};To=Co=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],To);var Eo;const Do=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),name:I.string().optional().describe(`Name for the trace (used in trace filename)`),screenshots:I.boolean().optional().describe(`Whether to capture screenshots during tracing (default: true)`),snapshots:I.boolean().optional().describe(`Whether to capture snapshots during tracing (default: true)`),sources:I.boolean().optional().describe(`Whether to include source files in the trace (default: false)`)});let Oo=class extends ${static{Eo=this}static TOOL_NAME=`browser_start_trace`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Do}getDefinition(){return{name:Eo.TOOL_NAME,description:`Starts Playwright performance tracing with configurable options for screenshots, snapshots, and sources. Use browser_stop_trace to stop and save the trace.`,inputSchema:I.toJSONSchema(Do,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId),n=t.context;if(!n)return this.error(`Page "${e.pageId}" is in extension mode and cannot start tracing`);let{name:r,screenshots:i=!0,snapshots:a=!0,sources:o=!1}=e;return await n.tracing.start({name:r,screenshots:i,snapshots:a,sources:o}),this.successJson({success:!0,message:`Tracing started`,pageId:t.id,options:{name:r??null,screenshots:i,snapshots:a,sources:o}})})}};Oo=Eo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Oo);var ko;const Ao=I.object({pageId:I.string().describe(`The page ID whose recording should be stopped.`),includeBase64:I.boolean().optional().describe(`Whether to include the base64-encoded WebM payload in the response.`)});let jo=class extends ${static{ko=this}static TOOL_NAME=`browser_stop_recording`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Ao}getDefinition(){return{name:ko.TOOL_NAME,description:`Stops an active extension/vm browser recording and returns the saved .webm path with optional base64 payload.`,inputSchema:I.toJSONSchema(Ao,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId);if(t.mode!==`extension`&&t.mode!==`vm`)return this.error(`Page "${e.pageId}" is in "${t.mode}" mode and cannot use extension recording`);let n=await this.browserService.stopPageRecording(t.browserId,t.id,{includeBase64:e.includeBase64});return this.successJson({stopped:!0,browserId:t.browserId,pageId:t.id,outputPath:n.outputPath,fileSizeBytes:n.fileSizeBytes,...n.videoBase64?{videoBase64:n.videoBase64}:{}})})}};jo=ko=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],jo);var Mo;const No=I.object({pageId:I.string().describe(`Optional page ID to override browser current page`),path:I.string().describe(`Absolute path where the trace file should be saved (e.g., "/Users/me/project/traces/trace.zip"). Must end with .zip extension.`)});let Po=class extends ${static{Mo=this}static TOOL_NAME=`browser_stop_trace`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return No}getDefinition(){return{name:Mo.TOOL_NAME,description:`Stops Playwright performance tracing and saves the trace file to a specified path. The trace can be viewed using Playwright Trace Viewer.`,inputSchema:I.toJSONSchema(No,{reused:`inline`})}}async execute(e){return this.safeExecute(async()=>{let t=this.resolvePageEntry(e.pageId),n=t.context;if(!n)return this.error(`Page "${e.pageId}" is in extension mode and cannot stop tracing`);let r=e.path;return x.isAbsolute(r)?r.endsWith(`.zip`)?(await n.tracing.stop({path:r}),this.successJson({success:!0,message:`Tracing stopped and saved`,pageId:t.id,tracePath:r,viewCommand:`npx playwright show-trace ${r}`})):this.error(`Trace path must end with .zip extension`):this.error(`Trace path must be an absolute path (e.g., /Users/me/project/traces/trace.zip)`)})}};Po=Mo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Po);var Fo;const Io=W.extend({pageId:I.string().describe(`Page ID to operate on`),text:I.string().describe(`The text to type character by character`),delay:I.number().optional().default(0).describe(`Delay between key presses in ms`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let Lo=class extends ${static{Fo=this}static TOOL_NAME=`browser_type`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return Io}getDefinition(){return{name:Fo.TOOL_NAME,description:`Types text into an element character by character, simulating real keyboard input. Does not clear existing content. Use browser_fill for replacing content.`,inputSchema:I.toJSONSchema(Io,{reused:`inline`})}}async execute(e){return this.executeWithMode(Fo.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId);return await(await this.elementLocator.locate(t,{selector:e.selector,xpath:e.xpath,uid:e.uid,frame:e.frame})).pressSequentially(e.text,{delay:e.delay,timeout:e.timeout}),this.success(`Typed: "${e.text}"`)})}};Lo=Fo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],Lo);var Ro;const zo=W.extend({pageId:I.string().describe(`Page ID to operate on`),files:I.union([I.string(),I.array(I.string())]).describe(`File path(s) to upload`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let Bo=class extends ${static{Ro=this}static TOOL_NAME=`browser_upload_file`;constructor(e,t,n,r){super(e,t,n),this.elementLocator=r}getInputSchema(){return zo}getDefinition(){return{name:Ro.TOOL_NAME,description:`Uploads file(s) to a file input element.`,inputSchema:I.toJSONSchema(zo,{reused:`inline`})}}async execute(e){return this.executeWithMode(Ro.TOOL_NAME,e.pageId,e,async()=>{let t=this.resolvePage(e.pageId);await(await this.elementLocator.locate(t,e)).setInputFiles(e.files,{timeout:e.timeout});let n=Array.isArray(e.files)?e.files.length:1;return this.success(`Uploaded ${n} file(s) successfully`)})}};Bo=Ro=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),B(3,_(R.ElementLocatorService)),z(`design:paramtypes`,[Object,Object,Object,Object])],Bo);var Vo;const Ho=I.object({pageId:I.string().describe(`Page ID to operate on`),type:I.enum([`element`,`text`,`timeout`,`loadState`]).optional().describe(`Type of wait condition`),selector:I.string().optional().describe(`CSS selector for element wait`),text:I.string().optional().describe(`Text to wait for (supports regex patterns)`),duration:I.number().optional().describe(`Duration in milliseconds for timeout wait`),state:I.enum([`load`,`domcontentloaded`,`networkidle`]).optional().describe(`Load state to wait for`),elementState:I.enum([`attached`,`detached`,`visible`,`hidden`]).optional().default(`visible`).describe(`Element state to wait for`),timeout:I.number().optional().default(H).describe(`Timeout in milliseconds`)});let Uo=class extends ${static{Vo=this}static TOOL_NAME=`browser_wait_for`;constructor(e,t,n){super(e,t,n)}getInputSchema(){return Ho}getDefinition(){return{name:Vo.TOOL_NAME,description:`Waits for a condition to be met: element visibility, text appearance, timeout duration, or page load state.`,inputSchema:I.toJSONSchema(Ho,{reused:`inline`})}}async execute(e){let t=e.type??(e.text?`text`:e.selector||e.elementState?`element`:e.duration===void 0?e.state?`loadState`:void 0:`timeout`);if(!t)return this.error(`Unable to infer wait type. Provide "type" or one of selector, text, duration, or state.`);if(t===`timeout`){let t=e.duration??1e3;return await new Promise(e=>setTimeout(e,t)),this.success(`Waited for ${t}ms`)}return this.executeWithMode(Vo.TOOL_NAME,e.pageId,{...e,type:t},async()=>{let n=this.resolvePage(e.pageId);switch(t){case`element`:if(!e.selector)throw Error(`selector is required for element wait`);return await n.locator(e.selector).waitFor({state:e.elementState??`visible`,timeout:e.timeout}),this.success(`Element "${e.selector}" is ${e.elementState??`visible`}`);case`text`:if(!e.text)throw Error(`text is required for text wait`);return await n.getByText(e.text).waitFor({state:`visible`,timeout:e.timeout}),this.success(`Text "${e.text}" appeared on page`);case`loadState`:{let t=e.state??`load`;return await n.waitForLoadState(t,{timeout:e.timeout}),this.success(`Page reached "${t}" state`)}default:throw Error(`Unknown wait type: ${t}`)}})}};Uo=Vo=V([v(),B(0,_(R.PageRegistry)),B(1,_(R.BrowserService)),B(2,_(R.ToolExecutor)),z(`design:paramtypes`,[Object,Object,Object])],Uo);const Wo=new g(e=>{e.bind(R.PortRegistryService).toDynamicValue(()=>new d(process.env.PORT_REGISTRY_PATH)).inSingletonScope(),e.bind(R.ProcessRegistryService).toDynamicValue(()=>new m(process.env.PROCESS_REGISTRY_PATH)).inSingletonScope(),e.bind(R.HttpServerHealthCheck).to(Ln).inSingletonScope(),e.bind(R.HttpServerManager).to(tr).inSingletonScope(),e.bind(R.McpPortAllocationService).to(sr).inSingletonScope(),e.bind(R.McpSessionTracker).to(cr).inSingletonScope()}),Go=new g(e=>{e.bind(R.ProfileService).to(fr).inSingletonScope(),e.bind(R.ProxyConfigService).to(X).inSingletonScope(),e.bind(R.PageRegistry).to(ur).inSingletonScope(),e.bind(R.BrowserService).to(Yt).inSingletonScope(),e.bind(R.BrowserLockManager).to(Ke).inSingletonScope(),e.bind(R.ChromeForTestingService).to(ct).inSingletonScope(),e.bind(R.ElementLocatorService).to(tn).inSingletonScope(),e.bind(R.PageMonitorService).to(lr).inSingletonScope(),e.bind(R.PauseController).to(dr).inSingletonScope(),e.bind(R.SpecBundlerService).to(Pr).inSingletonScope(),e.bind(R.SpecDiscoveryService).to(Fr).inSingletonScope(),e.bind(R.SpecMetadataService).to(Ir).inSingletonScope(),e.bind(R.SetupRunner).to(Dr).inSingletonScope(),e.bind(R.WebServerManager).to(Hr).inSingletonScope(),e.bind(R.SpecRunner).to(Rr).inSingletonScope(),e.bind(R.AutomationRunner).to(We).inSingletonScope(),e.bind(R.TelemetryService).to(G).inSingletonScope(),e.bind(R.CodeSnippetService).toDynamicValue(()=>new en(process.env.BROWSE_TOOL_SNIPPETS_DIR)).inSingletonScope(),e.bind(R.ExtensionTaskQueue).to(kn).inSingletonScope(),e.bind(R.ExtensionToolDelegator).to(Mn).inSingletonScope(),e.bind(R.ToolExecutor).to(Vr).inSingletonScope(),e.bind(R.ExtensionSessionRegistry).to(cn).inSingletonScope(),e.bind(R.ExtensionPageProxy).to(sn).inSingletonScope(),e.bind(R.ExtensionSpecRunner).to(ln).inSingletonScope(),e.bind(R.StealthLauncher).to(zr).inSingletonScope(),e.bind(R.HttpServerHealthCheck).to(Ln).inSingletonScope(),e.bind(R.HttpBrowserClient).to(In).inSingletonScope(),e.bind(R.RemoteToolExecutor).to(Er).inSingletonScope(),e.bind(R.McpPortAllocationService).to(sr).inSingletonScope(),e.bind(R.ProcessRegistryService).toDynamicValue(()=>new m(process.env.PROCESS_REGISTRY_PATH)).inSingletonScope(),e.bind(R.WebSocketHub).to(ai).inSingletonScope(),e.bind(R.IdleCleanupService).to(or).inSingletonScope()}),Ko=new g(e=>{e.bind(R.Tool).to(pi).inSingletonScope(),e.bind(R.Tool).to(Wi).inSingletonScope(),e.bind(R.Tool).to(Lo).inSingletonScope(),e.bind(R.Tool).to(yo).inSingletonScope(),e.bind(R.Tool).to(aa).inSingletonScope(),e.bind(R.Tool).to(Mi).inSingletonScope(),e.bind(R.Tool).to(qa).inSingletonScope(),e.bind(R.Tool).to(ca).inSingletonScope(),e.bind(R.Tool).to(Bo).inSingletonScope(),e.bind(R.Tool).to(Ra).inSingletonScope(),e.bind(R.Tool).to(Xi).inSingletonScope(),e.bind(R.Tool).to($i).inSingletonScope(),e.bind(R.Tool).to(Xa).inSingletonScope(),e.bind(R.Tool).to(li).inSingletonScope(),e.bind(R.Tool).to(Uo).inSingletonScope(),e.bind(R.Tool).to(So).inSingletonScope(),e.bind(R.Tool).to(po).inSingletonScope(),e.bind(R.Tool).to(Wa).inSingletonScope(),e.bind(R.Tool).to(ka).inSingletonScope(),e.bind(R.Tool).to(Va).inSingletonScope(),e.bind(R.Tool).to(go).inSingletonScope(),e.bind(R.Tool).to(yi).inSingletonScope(),e.bind(R.Tool).to(na).inSingletonScope(),e.bind(R.Tool).to(Ea).inSingletonScope(),e.bind(R.Tool).to(qi).inSingletonScope(),e.bind(R.Tool).to(Ca).inSingletonScope(),e.bind(R.Tool).to(Ri).inSingletonScope(),e.bind(R.Tool).to(Fi).inSingletonScope(),e.bind(R.Tool).to($a).inSingletonScope(),e.bind(R.Tool).to(Vi).inSingletonScope(),e.bind(R.Tool).to(To).inSingletonScope(),e.bind(R.Tool).to(jo).inSingletonScope(),e.bind(R.Tool).to(Oo).inSingletonScope(),e.bind(R.Tool).to(Po).inSingletonScope(),e.bind(R.Tool).to(Ma).inSingletonScope(),e.bind(R.Tool).to(Fa).inSingletonScope(),e.bind(R.Tool).to(Si).inSingletonScope(),e.bind(R.Tool).to(Ti).inSingletonScope(),e.bind(R.Tool).to(lo).inSingletonScope(),e.bind(R.Tool).to(oo).inSingletonScope(),e.bind(R.Tool).to(Oi).inSingletonScope(),e.bind(R.Tool).to(ba).inSingletonScope(),e.bind(R.Tool).to(gi).inSingletonScope(),e.bind(R.Tool).to(no).inSingletonScope()});function qo(){let e=new h({defaultScope:`Singleton`});return e.load(Wo),e}function Jo(){let e=new h({defaultScope:`Singleton`});return e.load(Go,Ko),e}const Yo=new g(e=>{e.bind(R.PortRegistryService).toDynamicValue(()=>new d(process.env.PORT_REGISTRY_PATH)).inSingletonScope(),e.bind(R.ProfileService).to(fr).inSingletonScope(),e.bind(R.ProxyConfigService).to(X).inSingletonScope(),e.bind(R.PageRegistry).to(ur).inSingletonScope(),e.bind(R.BrowserService).to(Yt).inSingletonScope(),e.bind(R.ElementLocatorService).to(tn).inSingletonScope(),e.bind(R.PageMonitorService).to(lr).inSingletonScope(),e.bind(R.PauseController).to(dr).inSingletonScope(),e.bind(R.SpecBundlerService).to(Pr).inSingletonScope(),e.bind(R.SpecDiscoveryService).to(Fr).inSingletonScope(),e.bind(R.SpecMetadataService).to(Ir).inSingletonScope(),e.bind(R.SetupRunner).to(Dr).inSingletonScope(),e.bind(R.WebServerManager).to(Hr).inSingletonScope(),e.bind(R.SpecRunner).to(Rr).inSingletonScope(),e.bind(R.AutomationRunner).to(We).inSingletonScope(),e.bind(R.HttpServerHealthCheck).to(Ln).inSingletonScope(),e.bind(R.HttpServerManager).to(tr).inSingletonScope(),e.bind(R.HttpBrowserClient).to(In).inSingletonScope(),e.bind(R.RemoteToolExecutor).to(Er).inSingletonScope(),e.bind(R.McpPortAllocationService).to(sr).inSingletonScope(),e.bind(R.ProcessRegistryService).toDynamicValue(()=>new m(process.env.PROCESS_REGISTRY_PATH)).inSingletonScope(),e.bind(R.ExtensionTaskQueue).to(kn).inSingletonScope(),e.bind(R.ExtensionToolDelegator).to(Mn).inSingletonScope(),e.bind(R.ToolExecutor).to(Vr).inSingletonScope(),e.bind(R.StealthLauncher).to(zr).inSingletonScope(),e.bind(R.BrowserLockManager).to(Ke).inSingletonScope(),e.bind(R.ExtensionSessionRegistry).to(cn).inSingletonScope(),e.bind(R.ExtensionPageProxy).to(sn).inSingletonScope(),e.bind(R.ExtensionSpecRunner).to(ln).inSingletonScope(),e.bind(R.McpSessionTracker).to(cr).inSingletonScope(),e.bind(R.ChromeForTestingService).to(ct).inSingletonScope(),e.bind(R.IdleCleanupService).to(or).inSingletonScope()}),Xo=new h({defaultScope:`Singleton`});Xo.load(Yo,Ko);function Zo(){let e=new h({defaultScope:`Singleton`});return e.load(Yo,Ko),e}var Qo=class extends Error{code=`TRANSPORT_CONNECTION_ERROR`;recovery=`Check that stdio streams are available and try again.`;constructor(e,t){super(e,t),this.name=`TransportConnectionError`}},$o=class{server;transport=null;constructor(e){this.server=e}async start(){let e=new Ve;try{await this.server.connect(e),this.transport=e,console.error(`browse-tool MCP server started on stdio`)}catch(t){try{await e.close()}catch{}let n=new Qo(`Failed to establish stdio transport connection`,{cause:t});throw console.error(`[${n.code}] ${n.message}. Recovery: ${n.recovery}`),n}}async stop(){this.transport&&=(await this.transport.close(),null)}},es=class extends Error{code=`TRANSPORT_CONNECTION_ERROR`;recovery=`Check that the HTTP host and port are available and try again.`;constructor(e,t){super(e,t),this.name=`TransportConnectionError`}};async function ts(e){let t=[];for await(let n of e)t.push(typeof n==`string`?Buffer.from(n):n);if(t.length===0)return;let n=Buffer.concat(t).toString(`utf8`).trim();if(n)return JSON.parse(n)}function ns(e,t,n){e.headersSent||(e.writeHead(t,{"Content-Type":`application/json`}),e.end(JSON.stringify(n)))}var rs=class{sessions=new Map;closingSessions=new Set;get(e){return this.sessions.get(e)}set(e,t){this.sessions.set(e,t)}has(e){return this.sessions.has(e)}remove(e){this.sessions.delete(e)}isClosing(e){return this.closingSessions.has(e)}async closeSession(e){let t=this.sessions.get(e);if(t){this.closingSessions.add(e),this.sessions.delete(e),t.server.close();try{await t.onClose?.()}finally{this.closingSessions.delete(e)}}}async clear(){let e=[...this.sessions.keys()];for(let t of e)await this.closeSession(t)}},is=class{sessionFactory;config;sessionManager=new rs;httpServer=null;constructor(e,t){this.sessionFactory=e,this.config={host:t.host,port:t.port,path:t.path??`/mcp`}}async createSession(e){let t=this.sessionFactory({headers:e.headers}),n=new Ue({sessionIdGenerator:()=>L(),enableJsonResponse:!0,onsessioninitialized:e=>{this.sessionManager.set(e,{server:t.server,transport:n,onClose:t.onClose})},onsessionclosed:async e=>{await this.sessionManager.closeSession(e)}});return n.onclose=()=>{n.sessionId&&(this.sessionManager.isClosing(n.sessionId)||this.sessionManager.remove(n.sessionId))},await t.server.connect(n),n}async handleMcpRequest(e,t){let n=e.headers[`mcp-session-id`],r=Array.isArray(n)?n[0]:n,i;if(e.method===`POST`)try{i=await ts(e)}catch(e){ns(t,400,{jsonrpc:`2.0`,error:{code:-32700,message:e instanceof Error?e.message:`Invalid JSON body`},id:null});return}let a;if(r?a=this.sessionManager.get(r)?.transport:e.method===`POST`&&Be(i)&&(a=await this.createSession(e)),!a){ns(t,r?404:400,{jsonrpc:`2.0`,error:{code:-32e3,message:r?`Session not found`:`Bad Request: No valid session ID provided`},id:null});return}await a.handleRequest(e,t,i)}async start(){if(this.httpServer)throw new es(`HTTP transport server is already running`);await new Promise((e,t)=>{let n=He((e,t)=>{(async()=>{try{let n=new URL(e.url??`/`,`http://${e.headers.host??this.config.host}`).pathname;if(n===`/health`&&e.method===`GET`){ns(t,200,{status:`ok`,transport:`streamable-http`});return}if(n!==this.config.path){ns(t,404,{error:`Not found`});return}await this.handleMcpRequest(e,t)}catch(e){ns(t,500,{jsonrpc:`2.0`,error:{code:-32603,message:(e instanceof Error?e:Error(String(e))).message},id:null})}})()});n.once(`error`,e=>{t(new es(`Failed to establish streamable HTTP transport connection`,{cause:e}))}),n.listen(this.config.port,this.config.host,()=>{this.httpServer=n,process.stderr.write(`browse-tool MCP server started on http://${this.config.host}:${this.config.port}${this.config.path}\n`),process.stderr.write(`Health check: http://${this.config.host}:${this.config.port}/health\n`),e()})})}async stop(){if(await this.sessionManager.clear(),!this.httpServer)return;let e=this.httpServer;this.httpServer=null,await new Promise((t,n)=>{e.close(e=>{if(e){n(e);return}t()})})}};export{Xe as C,Je as S,R as T,Tn as _,Jo as a,wt as b,Tr as c,Nn as d,q as f,G as g,kn as h,Zo as i,cr as l,Mn as m,$o as n,qo as o,K as p,Xo as r,pr as s,is as t,ir as u,sn as v,qe as w,nt as x,Ot as y};
|