@agimon-ai/browse-tool 0.15.2 → 0.16.0

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.mjs CHANGED
@@ -1,8 +1,8 @@
1
1
  #!/usr/bin/env node
2
- import{C as e,D as t,E as n,O as r,S as i,T as a,_ as o,a as s,b as c,c as l,d as u,f as d,g as f,h as p,i as m,l as h,m as g,n as _,o as v,p as y,s as b,t as ee,u as x,v as S,w as C,x as w,y as T}from"./streamable-http-CZoE_y6A.mjs";import{stripTypeScriptTypes as E}from"node:module";import"reflect-metadata/lite";import{DEFAULT_PORT_RANGE as D,PortRegistryService as O}from"@agimon-ai/foundation-port-registry";import{createProcessLease as k,resolveSiblingRegistryPath as te}from"@agimon-ai/foundation-process-registry";import{Container as A,ContainerModule as ne}from"inversify";import j from"node:path";import{existsSync as M,readFileSync as re,statSync as ie}from"node:fs";import{readFile as N,stat as ae}from"node:fs/promises";import{homedir as oe}from"node:os";import{z as P}from"zod";import{SpanStatusCode as F}from"@opentelemetry/api";import{randomUUID as se}from"node:crypto";import{Server as ce}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as le,ListToolsRequestSchema as ue}from"@modelcontextprotocol/sdk/types.js";import{StdioServerTransport as de}from"@modelcontextprotocol/sdk/server/stdio.js";import{Command as I,Option as fe}from"commander";import{serve as pe}from"@hono/node-server";import{Hono as L}from"hono";import{cors as me}from"hono/cors";import{createNodeWebSocket as he}from"@hono/node-ws";import{raw as ge}from"hono/html";import{Fragment as _e}from"hono/jsx";import{jsx as R,jsxs as z}from"hono/jsx/jsx-runtime";var ve=`0.15.1`;const ye=process.env.BROWSE_TOOL_DEBUG_EXTENSION_RECORDING===`1`;function be(e,t){if(!ye)return;let n=t?` ${JSON.stringify(t)}`:``;console.log(`[ExtensionRecordingDebug] ${e}${n}`)}function xe(e){let t=new L,n;try{n=e.get(r.TelemetryService)}catch{n=new f}return t.get(`/telemetry-config`,e=>{try{let t=new URL(e.req.url).origin;return e.json(o(process.env,t))}catch(t){return e.json({enabled:!1,error:t instanceof Error?t.message:String(t)},500)}}),t.get(`/tasks`,t=>{try{let n=e.get(r.ExtensionTaskQueue).getNextTask();return n?t.json({task:{id:n.id,tool:n.tool,arguments:n.arguments,telemetry:n.telemetry}}):t.json({})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/result`,async t=>{try{let n=await t.req.json();if(!n.taskId)return t.json({success:!1,error:`Missing taskId in request body`},400);let i=e.get(r.ExtensionTaskQueue),a=e.get(r.BrowserService),o={taskId:n.taskId,success:n.success,result:n.result,error:n.error},s=i.submitResult(o);return s?(s.browserId&&a.recordBrowserActivity(s.browserId,s.pageId),t.json({success:!0})):t.json({success:!1,error:`Task ${n.taskId} not found or already completed`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/status`,t=>{try{let n=e.get(r.ExtensionTaskQueue),i=n.getConnectionStatus(),a={connected:i.connected,lastPollAt:i.lastPollAt?.toISOString(),lastResultAt:i.lastResultAt?.toISOString(),pendingTasks:i.pendingTasks,queueSize:n.queueSize};return t.json(a)}catch(e){return t.json({connected:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/register`,async t=>{try{let n=await t.req.json();if(!n.browserId)return t.json({success:!1,error:`Missing browserId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).register(n);return t.json({success:!0,session:{id:i.id,browserId:i.browserId,controlMode:i.controlMode,createdAt:i.createdAt.toISOString()}})}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/heartbeat`,async t=>{try{let n=await t.req.json();if(!n.sessionId)return t.json({success:!1,error:`Missing sessionId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).heartbeat(n);return i?t.json({success:!0,session:{id:i.id,controlMode:i.controlMode,handoffRequested:i.handoffRequested,lastHeartbeatAt:i.lastHeartbeatAt.toISOString()}}):t.json({success:!1,error:`Session ${n.sessionId} not found`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/tab-mapped`,async t=>{try{let n=await t.req.json();if(!n.pageId||typeof n.tabId!=`number`)return t.json({success:!1,error:`Missing pageId or tabId in request body`},400);let i=e.get(r.PageRegistry),a=e.get(r.BrowserService),o=i.get(n.pageId);return o?(o.extensionTabId=n.tabId,a.recordBrowserActivity(o.browserId,n.pageId),t.json({success:!0})):t.json({success:!1,error:`Page ${n.pageId} not found`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/recording`,async t=>{try{let n=await t.req.json();if(!n.browserId)return t.json({success:!1,error:`Missing browserId in request body`},400);let i=e.get(r.BrowserService),a=await i.persistExtensionRecordingArtifact(n.browserId,n.videoBase64);return be(`artifact received`,{browserId:n.browserId,videoBase64Size:n.videoBase64?.length??0,persisted:a}),a?(i.recordBrowserActivity(n.browserId),t.json({success:!0})):t.json({success:!1,error:`Browser "${n.browserId}" has no active recording target`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/recording/chunk`,async t=>{try{let i=await t.req.json();if(!i.browserId||!i.chunkBase64)return t.json({success:!1,error:`Missing browserId or chunkBase64 in request body`},400);let a=e.get(r.BrowserService),o=await a.persistExtensionRecordingChunk(i.browserId,i.chunkBase64);return be(`chunk received`,{browserId:i.browserId,chunkIndex:i.chunkIndex,mimeType:i.mimeType,persisted:!!o,chunkBase64Size:i.chunkBase64.length}),o?(n.log(`debug`,`extension recording chunk received`,{attributes:{"browse_tool.extension.recording.chunk_received":!0,"browse_tool.browser.id":i.browserId,"browse_tool.recording.chunk_bytes":o.chunkBytes,"browse_tool.recording.total_bytes":o.totalBytes,"browse_tool.recording.chunk_count":o.chunkCount,...typeof i.chunkIndex==`number`?{"browse_tool.recording.chunk_index":i.chunkIndex}:{},...typeof i.mimeType==`string`?{"browse_tool.recording.mime_type":i.mimeType}:{}}}),a.recordBrowserActivity(i.browserId),t.json({success:!0})):t.json({success:!1,error:`Browser "${i.browserId}" has no active recording target`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/browser-log`,async e=>{try{let t=await e.req.json();return!t.message||typeof t.message!=`string`?e.json({success:!1,error:`Missing message in request body`},400):(n.log(t.level??`info`,t.message,{attributes:{"browse_tool.extension.log_relay":!0,...typeof t.attributes==`object`&&t.attributes!==null?t.attributes:{}}}),be(`browser log relayed`,{level:t.level??`info`,message:t.message,attributes:t.attributes}),e.json({success:!0}))}catch(t){return e.json({success:!1,error:t instanceof Error?t.message:String(t)},500)}}),t.post(`/handoff`,async t=>{try{let n=await t.req.json();if(!n.sessionId)return t.json({success:!1,error:`Missing sessionId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).requestHandoff(n);return i?t.json({success:!0,message:`Handoff requested. AI will take control when ready.`,session:{id:i.id,controlMode:i.controlMode,handoffRequested:i.handoffRequested}}):t.json({success:!1,error:`Session ${n.sessionId} not found`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/handoff/acknowledge`,async t=>{try{let n=await t.req.json();if(!n.sessionId)return t.json({success:!1,error:`Missing sessionId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).acknowledgeHandoff(n.sessionId);return i?t.json({success:!0,message:`Handoff acknowledged. AI now has control.`,session:{id:i.id,controlMode:i.controlMode,handoffRequested:i.handoffRequested}}):t.json({success:!1,error:`Session ${n.sessionId} not found or no handoff pending`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/sessions`,t=>{try{let n=e.get(r.ExtensionSessionRegistry).listSessions();return t.json({sessions:n.map(e=>({id:e.id,browserId:e.browserId,tabId:e.tabId,currentUrl:e.currentUrl,controlMode:e.controlMode,activeSpecPath:e.activeSpecPath,handoffRequested:e.handoffRequested,createdAt:e.createdAt.toISOString(),lastHeartbeatAt:e.lastHeartbeatAt.toISOString()}))})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t}function Se(e){if(e==null||e===``)return;let t=typeof e==`number`?e:typeof e==`string`?Number.parseInt(e,10):NaN;if(!Number.isInteger(t)||t<=0||t>65535)throw Error(`Invalid port: ${e}`);return t}const Ce=[`pnpm-workspace.yaml`,`nx.json`,`.git`],we=`browse-tool-chrome`,Te=`tool`,Ee=process.env.NODE_ENV||`development`,De=`127.0.0.1`;function Oe(e=process.cwd()){let t=j.resolve(e);for(;;){for(let e of Ce)if(M(j.join(t,e)))return t;let e=j.dirname(t);if(e===t)return process.cwd();t=e}}function ke(){return new O(process.env.PORT_REGISTRY_PATH)}async function Ae(e,t,n){let r=ke(),i=await r.reservePort({repositoryPath:e,serviceName:we,serviceType:Te,environment:Ee,preferredPort:t,portRange:n,pid:process.pid,host:De,force:!0,metadata:{transport:`stdio`,mode:`chrome-serve`}});if(!i.success||!i.record)throw Error(i.error||`Failed to reserve port ${t}`);let a=!1;return{port:i.record.port,release:async()=>{if(a)return;a=!0;let n=await r.releasePort({repositoryPath:e,serviceName:we,serviceType:Te,environment:Ee,pid:process.pid});if(!n.success&&!n.error?.includes(`No matching registry entry`))throw Error(n.error||`Failed to release port ${t}`)}}}function je(){return new ne(e=>{e.bind(r.ExtensionTaskQueue).to(p).inSingletonScope(),e.bind(r.ExtensionToolDelegator).to(g).inSingletonScope()})}function Me(e){let t=new ce({name:`browse-tool-chrome`,version:`0.1.0`},{capabilities:{tools:{}}}),n=e.getSupportedTools().map(e=>({name:e,description:`Browser automation tool (Chrome extension mode): ${e}`,inputSchema:{type:`object`,properties:{},additionalProperties:!0}}));return t.setRequestHandler(ue,async()=>({tools:n})),t.setRequestHandler(le,async t=>{let{name:n,arguments:r}=t.params;return await e.executeTool(n,r||{})}),t}const Ne=new I(`chrome-serve`).description(`[DEPRECATED] Start MCP server with Chrome extension HTTP polling for bot-detection-free browser automation`).option(`-p, --port <port>`,`HTTP server port for extension polling`).option(`-v, --verbose`,`Enable verbose output`,!1).option(`--wait-for-extension`,`Wait for extension to connect before accepting MCP requests`,!1).action(async e=>{console.error(``),console.error(`╔════════════════════════════════════════════════════════════════╗`),console.error(`║ DEPRECATED: chrome-serve is deprecated. ║`),console.error(`║ Use mcp-serve instead - extension routes are now ║`),console.error(`║ automatically available in the HTTP server at /extension/*. ║`),console.error(`╚════════════════════════════════════════════════════════════════╝`),console.error(``);let t,n;try{let i=Se(e.port),a=Oe(process.cwd()),o=process.env.PORT_REGISTRY_PATH;o&&(process.env.PROCESS_REGISTRY_PATH=te(o,`processes.json`)),t=await Ae(a,i??D.min,i?{min:i,max:i}:D);let s=t.port;n=await k({repositoryPath:a,serviceName:we,serviceType:Te,environment:Ee,pid:process.pid,port:s,host:De,command:process.argv[1],args:process.argv.slice(2),metadata:{transport:`stdio`,command:`chrome-serve`,waitForExtension:e.waitForExtension}}),e.verbose&&(console.error(`Chrome Extension MCP Server starting...`),console.error(` HTTP Port: ${s}`),console.error(` Wait for extension: ${e.waitForExtension}`));let c=new A({defaultScope:`Singleton`});c.load(je());let l=c.get(r.ExtensionTaskQueue),u=c.get(r.ExtensionToolDelegator),d=new L;d.use(`*`,me({origin:e=>e??null,credentials:!0,allowMethods:[`GET`,`POST`,`OPTIONS`],allowHeaders:[`Content-Type`,`Accept`]}));let f=xe(c);d.route(`/extension`,f),d.get(`/health`,e=>{let t=l.getConnectionStatus();return e.json({status:`healthy`,service:`browse-tool-chrome`,extension:t})});let p=pe({fetch:d.fetch,port:s});p.on(`error`,e=>{e.code===`EADDRINUSE`?(console.error(`Error [PORT_IN_USE]: Port ${s} is already in use.`),console.error(`Recovery: Try a different port with --port <port>`)):console.error(`Error [SERVER_ERROR]: HTTP server error: ${e.message}`),process.exit(1)}),console.error(`HTTP server started on port ${s}`),console.error(`Waiting for Chrome extension to connect...`),console.error(`Extension should poll: http://localhost:${s}/extension/tasks`),e.waitForExtension&&(console.error(`Waiting for extension connection before starting MCP...`),await new Promise(e=>{let t=setInterval(()=>{l.getConnectionStatus().connected&&(clearInterval(t),console.error(`Extension connected!`),e())},500)}));let m=Me(u),h=new de;await m.connect(h),console.error(`Chrome extension MCP server started on stdio`);let g=async e=>{console.error(`\nReceived ${e}, shutting down gracefully...`);try{l.clearAllTasks(`Server shutting down`),await h.close(),p.close(),await n.release({kill:!1}),await t.release(),process.exit(0)}catch(e){console.error(`Error during shutdown:`,e),process.exit(1)}};process.on(`SIGINT`,()=>g(`SIGINT`)),process.on(`SIGTERM`,()=>g(`SIGTERM`))}catch(e){if(n||t)try{await n?.release(),await t?.release()}catch{}let r=e instanceof Error?e.message:String(e);console.error(`Error [SERVER_ERROR]: Failed to start Chrome extension MCP server: ${r}`),console.error(`Recovery: Check that the port is available and try again.`),process.exit(1)}}),Pe=`BROWSE_TOOL_CONFIG`,Fe=`.browse-tool`,Ie=`config.json`,Le=[`json`,`text`,`quiet`],Re={commands:{},tools:{}},ze=P.record(P.string(),P.unknown()),Be=P.object({mcpServe:P.object({type:P.string().optional(),browser:P.string().optional(),headless:P.boolean().optional(),profile:P.string().optional(),mode:P.string().optional(),host:P.string().optional(),port:P.coerce.number().int().positive().optional(),httpPort:P.coerce.number().int().positive().optional(),tags:P.string().optional(),exclude:P.string().optional(),customTools:P.string().optional(),chromeForTestingPath:P.string().optional(),snippetsDir:P.string().optional(),registryPath:P.string().optional(),registryDir:P.string().optional(),pidsDir:P.string().optional(),profilesDir:P.string().optional(),proxyConfigDir:P.string().optional()}).partial().optional(),httpServe:P.object({port:P.coerce.number().int().positive().optional(),headless:P.boolean().optional(),idleTimeout:P.coerce.number().positive().optional(),host:P.string().optional(),registryDir:P.string().optional(),registryPath:P.string().optional(),pidsDir:P.string().optional(),profilesDir:P.string().optional(),snippetsDir:P.string().optional(),proxyConfigDir:P.string().optional()}).partial().optional(),exec:P.object({format:P.enum(Le).optional(),color:P.boolean().optional(),port:P.coerce.number().int().positive().optional()}).partial().optional(),tools:P.object({format:P.enum(Le).optional(),color:P.boolean().optional(),port:P.coerce.number().int().positive().optional()}).partial().optional(),status:P.record(P.string(),P.unknown()).optional(),stop:P.record(P.string(),P.unknown()).optional()}),Ve=P.object({commands:Be.default({}),tools:P.record(P.string(),ze).default({})});let B={config:Re};function He(e){for(let t=0;t<e.length;t+=1){let n=e[t];if(n===`--config`)return e[t+1];if(n.startsWith(`--config=`))return n.slice(9)}}function Ue(e){let t=j.resolve(e);if(!M(t))throw Error(`Config path not found: ${t}`);return ie(t).isDirectory()?j.join(t,Ie):t}function We(e,t){let n=t.env??process.env,r=t.cwd??process.cwd(),i=t.homeDir??oe(),a=He(e);if(a)return Ue(a);if(n[Pe])return Ue(n[Pe]);let o=j.join(r,Fe,Ie);if(M(o))return o;let s=j.join(i,Fe,Ie);if(M(s))return s}function Ge(e){let t=re(e,`utf8`),n=JSON.parse(t);return Ve.parse(n)}function Ke(e,t={}){let n=We(e,t);return n?{configPath:n,config:Ge(n)}:{config:Re}}function qe(e,t={}){return B=Ke(e,t),B}function V(e){return(B.config.commands??{})[e]??{}}function Je(e){return B.config.tools?.[e]??{}}function H(e,t,n,r,i){let a=e.getOptionValueSourceWithGlobals(t);return a!==void 0&&a!==`default`&&a!==`implied`?n:i===void 0?r===void 0?n:r:i}var Ye=class{port;exactPort;timeout;serverPort=null;constructor(e={}){this.port=e.port??u,this.exactPort=e.exactPort??!1,this.timeout=e.timeout??6e4}async ensureServer(){if(this.serverPort!==null)return this.serverPort;let e=await v().get(r.HttpServerManager).ensureRunning(this.port,{exactPort:this.exactPort});if(!e.running||!e.port)throw Error(e.error??`Failed to start HTTP server. Try running "browse-tool http-serve" manually.`);return this.serverPort=e.port,e.port}async getBaseUrl(){return`http://localhost:${await this.ensureServer()}`}async listTools(){let e=await this.getBaseUrl(),t=new AbortController,n=setTimeout(()=>t.abort(),this.timeout);try{let n=await fetch(`${e}/tools`,{method:`GET`,headers:{Accept:`application/json`},signal:t.signal});if(!n.ok)throw Error(`HTTP ${n.status}: ${n.statusText}`);let r=await n.json();if(r.error)throw Error(r.error);return r.tools}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(n)}}async listCustomTools(e){let t=await this.getBaseUrl(),n=new AbortController,r=setTimeout(()=>n.abort(),this.timeout);try{let r=new URL(`/custom-tools`,t);r.searchParams.set(`dir`,e);let i=await fetch(r,{method:`GET`,headers:{Accept:`application/json`},signal:n.signal});if(!i.ok)throw Error(`HTTP ${i.status}: ${i.statusText}`);let a=await i.json();if(a.error)throw Error(a.error);return a.tools}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(r)}}async listBrowsers(){let e=await this.getBaseUrl(),t=new AbortController,n=setTimeout(()=>t.abort(),this.timeout);try{let n=await fetch(`${e}/browsers`,{method:`GET`,headers:{Accept:`application/json`},signal:t.signal});if(!n.ok)throw Error(`HTTP ${n.status}: ${n.statusText}`);let r=await n.json();if(r.error)throw Error(r.error);return r.browsers}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(n)}}async execute(e,t){let n=await this.getBaseUrl(),r=e===`run_spec`?void 0:this.timeout,i=new AbortController,a=r===void 0?void 0:setTimeout(()=>i.abort(),r);try{let a=await fetch(`${n}/execute`,{method:`POST`,headers:{"Content-Type":`application/json`,Accept:`application/json`},body:JSON.stringify({tool:e,arguments:t}),signal:r===void 0?void 0:i.signal});if(!a.ok){let e=await a.text();throw Error(`HTTP ${a.status}: ${e||a.statusText}`)}let o=await a.json();return o.success?o.result??{content:[{type:`text`,text:`No result returned`}]}:{content:[{type:`text`,text:o.error??`Unknown error`}],isError:!0}}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{a&&clearTimeout(a)}}async executeCustomTool(e,t,n){let r=await this.getBaseUrl(),i=new AbortController,a=setTimeout(()=>i.abort(),this.timeout);try{let a=new URL(`/custom-tools`,r);a.searchParams.set(`dir`,e);let o=await fetch(a,{method:`POST`,headers:{"Content-Type":`application/json`,Accept:`application/json`},body:JSON.stringify({tool:t,arguments:n}),signal:i.signal});if(!o.ok){let e=await o.text();throw Error(`HTTP ${o.status}: ${e||o.statusText}`)}let s=await o.json();return s.success?s.result??{content:[{type:`text`,text:`No result returned`}]}:{content:[{type:`text`,text:s.error??`Unknown error`}],isError:!0}}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(a)}}wrapConnectionError(e){return e instanceof Error?e.message.includes(`ECONNREFUSED`)||e.message.includes(`fetch failed`)?Error(`Cannot connect to browse-tool HTTP server.\n\nTry one of the following:\n 1. Start the server: browse-tool http-serve\n 2. Check if another process is using port ${this.port}\n 3. Run with a different port: browse-tool --port 3201 <command>\n\nOriginal error: ${e.message}`,{cause:e}):e:Error(String(e),{cause:e})}};function U(e){return new Ye(e)}const Xe={reset:`\x1B[0m`,red:`\x1B[31m`,green:`\x1B[32m`,yellow:`\x1B[33m`,blue:`\x1B[34m`,magenta:`\x1B[35m`,cyan:`\x1B[36m`,gray:`\x1B[90m`,bold:`\x1B[1m`};function W(e,t,n){return n?`${Xe[t]}${e}${Xe.reset}`:e}function Ze(e,t){let{format:n,color:r}=t;return n===`quiet`?``:n===`json`?Qe(e):$e(e,r)}function Qe(e){return JSON.stringify(e,null,2)}function $e(e,t){let n=[];e.isError&&n.push(W(`Error:`,`red`,t));for(let r of e.content)r.type===`text`?n.push(et(r,t)):r.type===`image`?n.push(rt(r,t)):n.push(W(`[Unknown content type: ${r.type}]`,`yellow`,t));return n.join(`
3
- `)}function et(e,t){let n=e.text;try{return tt(JSON.parse(n),t)}catch{return n}}function tt(e,t){if(typeof e!=`object`||!e)return String(e);let n=[];for(let[r,i]of Object.entries(e)){let e=W(r,`cyan`,t),a=nt(i,t);n.push(`${e}: ${a}`)}return n.join(`
4
- `)}function nt(e,t){return e===null?W(`null`,`gray`,t):e===void 0?W(`undefined`,`gray`,t):typeof e==`boolean`?W(String(e),e?`green`:`red`,t):typeof e==`number`?W(String(e),`yellow`,t):typeof e==`string`?e.startsWith(`http://`)||e.startsWith(`https://`)?W(e,`blue`,t):e:Array.isArray(e)?e.length===0?W(`[]`,`gray`,t):JSON.stringify(e,null,2):typeof e==`object`?JSON.stringify(e,null,2):String(e)}function rt(e,t){let{mimeType:n,data:r}=e;return W(`[Image: ${n}, ~${Math.round(r.length*3/4/1024)}KB base64 data]`,`magenta`,t)}function G(e,t){return W(`Error: ${e instanceof Error?e.message:e}`,`red`,t)}function it(e,t){let n=Math.max(...e.map(e=>e.name.length)),r=[];for(let i of e){let e=W(i.name.padEnd(n),`cyan`,t);r.push(` ${e} ${i.description}`)}return r.join(`
5
- `)}function at(e,t){let n=V(`tools`),r=H(e,`format`,t.format,n.format),i=H(e,`color`,t.color,n.color);return{port:H(e,`port`,t.port,n.port===void 0?void 0:String(n.port),process.env.PLAYWRIGHT_PORT),formatterOptions:{format:r,color:i}}}const ot=new I(`list-custom-tools`).description(`List custom tools from a tools directory`).argument(`<dir>`,`Path to the tools directory containing tools.yaml`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u)).action(async function(e,t){let{port:n,formatterOptions:r}=at(this,t);try{let t=await U({port:Number.parseInt(n,10)}).listCustomTools(e),i=r.format===`text`?it(t,r.color):r.format===`quiet`?``:JSON.stringify(t,null,2);i&&console.log(i)}catch(e){console.error(G(e instanceof Error?e:String(e),r.color)),process.exit(1)}}),st=new I(`exec-custom-tool`).description(`Execute a custom tool from a tools directory`).argument(`<dir>`,`Path to the tools directory containing tools.yaml`).argument(`<tool>`,`Custom tool name to execute`).argument(`[args]`,`JSON arguments for the tool`,`{}`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u)).action(async function(e,t,n,r){let{port:i,formatterOptions:a}=at(this,r);try{let r;try{r=JSON.parse(n)}catch{console.error(G(`Invalid JSON arguments: ${n}`,a.color)),process.exit(1);return}let o=await U({port:Number.parseInt(i,10)}).executeCustomTool(e,t,r),s=Ze(o,a);s&&console.log(s),o.isError&&process.exit(1)}catch(e){console.error(G(e instanceof Error?e:String(e),a.color)),process.exit(1)}});new I(`custom-tools`).description(`List and execute custom tools through the HTTP server`).addCommand(ot).addCommand(st);const ct=new I(`docker-build-cft`).description(`Build the Chrome for Testing Docker image used by vm mode`).option(`--cft-version <version>`,`Chrome for Testing version to build`,t).option(`--image <image>`,`Docker image tag to produce`,a).option(`--platform <platform>`,`Docker target platform`,n).action(async t=>{try{let n=await i({version:t.cftVersion,image:t.image,platform:t.platform,stdio:`inherit`});console.log(`Built Docker image ${n.image}`),console.log(` Version: ${n.version}`),console.log(` Platform: ${n.platform}`),console.log(` Archive: ${e(n.platform)}`)}catch(e){console.error(e instanceof Error?e.message:String(e)),process.exit(1)}}),lt=new I(`exec`).description(`Execute a tool directly with JSON arguments`).argument(`<tool>`,`Tool name to execute (e.g., browser_launch)`).argument(`[args]`,`JSON arguments for the tool`,`{}`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u)).action(async function(e,t,n){let r=V(`exec`),i=H(this,`format`,n.format,r.format),a=H(this,`color`,n.color,r.color),o=H(this,`port`,n.port,r.port===void 0?void 0:String(r.port),process.env.PLAYWRIGHT_PORT),s=this.getOptionValueSourceWithGlobals(`port`),c=s!==void 0&&s!==`default`&&s!==`implied`,l={format:i,color:a};try{let n;try{n=JSON.parse(t)}catch{console.error(G(`Invalid JSON arguments: ${t}`,l.color)),process.exit(1)}let r=await U({port:Number.parseInt(o,10),exactPort:c}).execute(e,n),i=Ze(r,l);i&&console.log(i),r.isError&&process.exit(1)}catch(e){console.error(G(e instanceof Error?e:String(e),l.color)),process.exit(1)}}),K=`pageId`,q=`browserId`,ut=P.record(P.string(),P.unknown()),dt=P.object({type:P.literal(`object`),properties:P.record(P.string(),P.unknown()).optional(),required:P.array(P.string()).optional(),additionalProperties:P.boolean().optional()}).passthrough(),ft=P.object({name:P.string().min(1),description:P.string().min(1),script:P.string().min(1),suggestionActions:P.string().min(1).optional(),capabilities:ut,inputSchema:dt}),pt=P.object({tools:P.array(ft)}),mt=process.env.BROWSE_TOOL_DEBUG_CUSTOM_TOOLS===`1`;function ht(e){return JSON.stringify(e,(e,t)=>typeof t==`string`&&t.length>240?`${t.slice(0,240)}...<trimmed>`:t)}function gt(e,t){if(mt){if(t){console.error(`[CustomToolService] ${e}`,t);return}console.error(`[CustomToolService] ${e}`)}}function J(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}function _t(e){return J(e)&&Array.isArray(e.content)}function Y(e){let t=e.trim();return t===``?``:t.startsWith(`"`)&&t.endsWith(`"`)||t.startsWith(`[`)&&t.endsWith(`]`)||t.startsWith(`{`)&&t.endsWith(`}`)?JSON.parse(t):t.startsWith(`'`)&&t.endsWith(`'`)?t.slice(1,-1):t===`true`?!0:t===`false`?!1:t===`null`?null:/^-?\d+(\.\d+)?$/.test(t)?Number(t):t}function vt(e){if(!e||Object.keys(e).length===0)return;let t={};for(let[n,r]of Object.entries(e))if(r!=null){if(typeof r==`string`||typeof r==`number`||typeof r==`boolean`){t[n]=r;continue}t[n]=JSON.stringify(r)}return Object.keys(t).length>0?t:void 0}function yt(e){return e.replace(/^\uFEFF/,``).split(/\r?\n/).map(e=>{if(e.includes(` `))throw Error(`Tab indentation is not supported in tools.yaml`);let t=e.replace(/\s+#.*$/,``);return t.trim().length===0?null:{indent:t.match(/^ */)?.[0].length??0,text:t.trim()}}).filter(e=>e!==null)}function X(e,t,n){let r=e[t];if(!r||r.indent!==n)throw Error(`Invalid indentation in tools.yaml at line ${t+1}`);return r.text.startsWith(`-`)?bt(e,t,n):xt(e,t,n)}function bt(e,t,n){let r=[],i=t;for(;i<e.length;){let t=e[i];if(t.indent<n||t.indent!==n||!t.text.startsWith(`-`))break;let a=t.text.slice(1).trim();if(a===``){let t=e[i+1];if(!t||t.indent<=n){r.push(null),i+=1;continue}let[a,o]=X(e,i+1,t.indent);r.push(a),i=o;continue}if(a.includes(`:`)){let[t,o]=St(a),s={};if(o===void 0){let r=e[i+1];if(!r||r.indent<=n)throw Error(`Expected nested value for "${t}" in tools.yaml`);let[a,o]=X(e,i+1,r.indent);s[t]=a,i=o}else s[t]=Y(o),i+=1;for(;i<e.length&&e[i].indent>n;){let t=e[i];if(t.indent!==n+2||t.text.startsWith(`-`)){let[n,r]=X(e,i,t.indent);if(!J(n))throw Error(`Expected object entry in tools.yaml at line ${i+1}`);Object.assign(s,n),i=r;continue}let[r,a]=St(t.text);if(a===void 0){let n=e[i+1];if(!n||n.indent<=t.indent)throw Error(`Expected nested value for "${r}" in tools.yaml`);let[a,o]=X(e,i+1,n.indent);s[r]=a,i=o;continue}s[r]=Y(a),i+=1}r.push(s);continue}r.push(Y(a)),i+=1}return[r,i]}function xt(e,t,n){let r={},i=t;for(;i<e.length;){let t=e[i];if(t.indent<n||t.indent!==n||t.text.startsWith(`-`))break;let[a,o]=St(t.text);if(o===void 0){let t=e[i+1];if(!t||t.indent<=n){r[a]=null,i+=1;continue}let[o,s]=X(e,i+1,t.indent);r[a]=o,i=s;continue}r[a]=Y(o),i+=1}return[r,i]}function St(e){let t=e.indexOf(`:`);if(t===-1)throw Error(`Invalid tools.yaml entry: "${e}"`);let n=e.slice(0,t).trim(),r=e.slice(t+1).trim();return[n,r===``?void 0:r]}function Ct(e){let t=yt(e);if(t.length===0)return{};let[n]=X(t,0,t[0].indent);return n}function wt(e){e.inputSchema.properties===void 0&&(e.inputSchema.properties={});let t=e.inputSchema.properties;if(!J(t))throw Error(`Custom tool "${e.name}" inputSchema.properties must be an object`);let n=t[K],r=t[q];if(n!==void 0&&!(J(n)&&n.type===`string`))throw Error(`Custom tool "${e.name}" must define inputSchema.properties.pageId as type "string"`);if(r!==void 0&&!(J(r)&&r.type===`string`))throw Error(`Custom tool "${e.name}" must define inputSchema.properties.browserId as type "string"`);n===void 0&&(t[K]={type:`string`,description:`Browse-tool page ID to run this custom tool against. Either pageId or browserId is required at call time.`}),r===void 0&&(t[q]={type:`string`,description:`Browse-tool browser ID to run this custom tool against when no pageId is supplied. The tool will use the browser's current page or open a new one.`})}async function Tt(e){let t=E(await N(e,`utf8`),{mode:`strip`});return await import(`data:text/javascript;base64,${Buffer.from(t,`utf8`).toString(`base64`)}`)}function Et(e,t){let n=typeof t.run==`function`?t.run:void 0;if(!n)throw Error(`Custom tool script "${e}" must export a "run" function`);return{execute:n}}function Dt(e,t){return typeof e.suggestionActions==`string`&&e.suggestionActions.trim().length>0?e:{...e,suggestionActions:t}}function Ot(e,t){let n=e.content[0];if(n?.type===`text`&&typeof n.text==`string`)try{let r=JSON.parse(n.text);if(J(r)){let i=Dt(r,t);return{...e,content:[{...n,text:JSON.stringify(i,null,2)},...e.content.slice(1)]}}}catch{}return{...e,content:[...e.content,{type:`text`,text:`suggestionActions: ${t}`}]}}function kt(e,t,n){if(_t(t))return n?Ot(t,n):t;if(typeof t==`string`)return n?{content:[{type:`text`,text:JSON.stringify({result:t,suggestionActions:n},null,2)}]}:{content:[{type:`text`,text:t}]};if(t===void 0)return n?{content:[{type:`text`,text:JSON.stringify({result:`Custom tool "${e}" completed successfully`,suggestionActions:n},null,2)}]}:{content:[{type:`text`,text:`Custom tool "${e}" completed successfully`}]};let r=J(t)&&n?Dt(t,n):t;return{content:[{type:`text`,text:JSON.stringify(r,null,2)}]}}var At=class{constructor(e,t,n=new f,r){this.pageRegistry=e,this.extensionTaskQueue=t,this.telemetry=n,this.browserService=r}resolveToolPage(e,t,n){if(n.page)return n.page;if(n.mode===`extension`&&this.extensionTaskQueue){let e=new S(this.extensionTaskQueue);return e.setTarget(t,n.browserId),e}throw Error(`Custom tool "${e}" requires a supported page context`)}toPageSummary(e,t){return{pageId:t.id,url:t.url,title:t.title,active:e===t.id}}async createPageForBrowser(e,t,n){if(!this.browserService)throw Error(`Custom tool "${e}" requires browser service support to create a page`);let r=this.browserService.getBrowser(t);if(!r)throw Error(`Browser "${t}" not found`);let i=n?.setAsCurrent!==!1;if(r.mode===`extension`||r.mode===`vm`){if(!this.extensionTaskQueue)throw Error(`Custom tool "${e}" requires extension task support to create a page`);let a=this.pageRegistry.registerExtensionPage(t,void 0,n?.url,!1);try{let o=await this.extensionTaskQueue.queueTask(`browser_new_page`,{browserId:t,pageId:a,url:n?.url,setAsCurrent:i},1e4,t);if(!o.success)throw Error(o.error??`Custom tool "${e}" failed to create a page`);let s=o.result?.content[0]?.type===`text`?o.result.content[0].text:void 0,c={};if(typeof s==`string`&&s.length>0)try{c=JSON.parse(s)}catch{c={}}let l=this.pageRegistry.get(a);if(!l)throw Error(`Page "${a}" was not registered`);l.url=c.url??l.url,l.title=c.title??l.title,l.extensionTabId=c.tabId??l.extensionTabId;let u=await this.waitForResolvedPageMetadata(a);return r.pageIds.add(a),(i||!r.currentPageId)&&this.browserService.setCurrentPage(t,a),this.browserService.recordBrowserActivity(t,a),{...this.toPageSummary(r.currentPageId,u),page:this.resolveToolPage(e,a,u)}}catch(e){throw this.pageRegistry.remove(a),e}}let{pageId:a,page:o}=await this.browserService.newPage(t);n?.url&&(await o.goto(n.url),await this.pageRegistry.updateMetadata(a)),i&&this.browserService.setCurrentPage(t,a),this.browserService.recordBrowserActivity(t,a);let s=this.pageRegistry.get(a);if(!s)throw Error(`Page "${a}" was not registered`);return{...this.toPageSummary(r.currentPageId,s),page:this.resolveToolPage(e,a,s)}}async waitForResolvedPageMetadata(e){let t=Date.now(),n=this.pageRegistry.get(e);for(;n&&Date.now()-t<1500;){let t=typeof n.url==`string`&&n.url.length>0,r=typeof n.title==`string`&&n.title.length>0&&n.title!==`Extension Tab`;if(t&&r)return n;await new Promise(e=>setTimeout(e,50)),n=this.pageRegistry.get(e)}if(!n)throw Error(`Page "${e}" was not registered`);return n}async resolveExecutionContext(e,t){let n=typeof t[K]==`string`&&t[K].length>0?t[K]:void 0,r=typeof t[q]==`string`&&t[q].length>0?t[q]:void 0;if(!n&&!r)throw Error(`Custom tool "${e}" requires a string pageId or browserId`);if(n){let t=this.pageRegistry.get(n);if(!t)throw Error(`Page "${n}" not found`);let i=c();if(i&&t.ownerId&&t.ownerId!==i)throw Error(`Custom tool "${e}" cannot use page "${n}" owned by another session`);if(r&&t.browserId!==r)throw Error(`Custom tool "${e}" received pageId "${n}" for browser "${t.browserId}", not "${r}"`);let a=this.createBrowserHelper(e,t.browserId);return{pageId:n,pageEntry:t,page:this.resolveToolPage(e,n,t),browser:a}}let i=this.createBrowserHelper(e,r);try{let e=await i.getCurrentPage(),t=this.pageRegistry.get(e.pageId);if(!t)throw Error(`Page "${e.pageId}" not found`);return{pageId:e.pageId,pageEntry:t,page:e.page,browser:i}}catch(e){if(!(e instanceof Error?e.message:String(e)).includes(`has no pages`))throw e;let t=await i.newPage(),n=this.pageRegistry.get(t.pageId);if(!n)throw Error(`Page "${t.pageId}" not found`,{cause:e});return{pageId:t.pageId,pageEntry:n,page:t.page,browser:i}}}createBrowserHelper(e,t){return{browserId:t,mode:this.browserService?.getBrowser(t)?.mode??`extension`,listPages:async()=>{let e=this.browserService?.getBrowser(t);if(!e)throw Error(`Browser "${t}" not found`);return this.pageRegistry.findByBrowser(t).map(t=>this.toPageSummary(e.currentPageId,t))},getPage:async n=>{let r=this.pageRegistry.get(n);if(!r||r.browserId!==t)throw Error(`Page "${n}" not found in browser "${t}"`);let i=this.browserService?.getBrowser(t);return{...this.toPageSummary(i?.currentPageId??null,r),page:this.resolveToolPage(e,n,r)}},getCurrentPage:async()=>{let n=this.browserService?.getBrowser(t);if(!n)throw Error(`Browser "${t}" not found`);let r=c(),i;if(r){let e=this.pageRegistry.findByOwner(r).filter(e=>e.browserId===t);if(i=e[e.length-1]?.id,!i)throw Error(`Browser "${t}" has no pages owned by this session`)}else i=n.currentPageId??this.pageRegistry.findByBrowser(t)[0]?.id??void 0;if(!i)throw Error(`Browser "${t}" has no pages`);let a=this.pageRegistry.get(i);if(!a)throw Error(`Page "${i}" not found`);return{...this.toPageSummary(n.currentPageId,a),page:this.resolveToolPage(e,i,a)}},newPage:async n=>this.createPageForBrowser(e,t,n)}}createToolLogger(e,t,n){let r={"browse_tool.tool.name":e,"browse_tool.page.id":t,"browse_tool.browser.id":n.browserId,"browse_tool.execution.mode":n.mode},i=(e,t,n)=>{this.telemetry.log(e,t,{attributes:vt({...r,...n?.attributes??{}}),exception:n?.exception})};return{getTraceContext:()=>this.telemetry.getActiveTraceContext(),trace:(e,t)=>i(`trace`,e,t),debug:(e,t)=>i(`debug`,e,t),info:(e,t)=>i(`info`,e,t),warn:(e,t)=>i(`warn`,e,t),error:(e,t)=>i(`error`,e,t),fatal:(e,t)=>i(`fatal`,e,t)}}async listTools(e){return(await this.loadTools(e)).map(({name:e,description:t,suggestionActions:n,inputSchema:r,capabilities:i})=>({name:e,description:t,suggestionActions:n,inputSchema:r,capabilities:i}))}async executeTool(e,t,n){let r=typeof n[K]==`string`?n[K]:void 0,i=typeof n[q]==`string`?n[q]:void 0;return this.telemetry.runInSpan(`browse_tool.custom_tool.execute`,{attributes:{"browse_tool.tool.name":t,"browse_tool.custom_tools.directory":j.resolve(e),"browse_tool.page.id":r,"browse_tool.browser.id":i}},async r=>{let i=(await this.loadTools(e)).find(e=>e.name===t);if(!i)throw r?.setStatus({code:F.ERROR,message:`Custom tool "${t}" not found`}),Error(`Custom tool "${t}" not found`);let{pageId:a,pageEntry:o,page:s,browser:c}=await this.resolveExecutionContext(t,n),l=this.createToolLogger(t,a,o);r?.setAttributes({"browse_tool.browser.id":o.browserId,"browse_tool.execution.mode":o.mode,"browse_tool.page.id":a}),gt(`Executing custom tool`,{toolName:t,directory:j.resolve(e),pageId:a,browserId:o.browserId,mode:o.mode,input:ht(n),scriptPath:i.scriptPath});let u;try{u=await i.execute?.({page:s,browser:c,input:n,logger:l})}catch(e){let n=e instanceof Error?e.message:String(e);throw gt(`Custom tool execution failed`,{toolName:t,pageId:a,browserId:o.browserId,mode:o.mode,error:n,stack:e instanceof Error?e.stack:void 0}),Error(`Custom tool "${t}" failed on page "${a}": ${n}`,{cause:e})}gt(`Custom tool execution completed`,{toolName:t,pageId:a,browserId:o.browserId,mode:o.mode,result:ht(u)});let d=kt(t,u,i.suggestionActions),f=d.isError?d.content[0]?.text:void 0;return f&&r?.setStatus({code:F.ERROR,message:f}),d})}async loadTools(e){let t=j.resolve(e),n=j.join(t,`tools.yaml`),r=Ct(await N(n,`utf8`).catch(e=>{throw Error(`Failed to read custom tool manifest at "${n}": ${e instanceof Error?e.message:String(e)}`)})),i=pt.parse(r);return Promise.all(i.tools.map(async e=>{wt(e);let n=j.resolve(t,e.script);await ae(n).catch(t=>{throw Error(`Custom tool "${e.name}" script not found at "${n}": ${t instanceof Error?t.message:String(t)}`)});let r=Et(n,await Tt(n));return{name:e.name,description:e.description,suggestionActions:e.suggestionActions,inputSchema:e.inputSchema,capabilities:e.capabilities,scriptPath:n,...r}}))}};function jt(e){let t=new L;return t.get(`/browsers`,async t=>{try{let n=e.get(r.BrowserService),i=e.get(r.PageRegistry),a=n.listBrowsers(),o=i.list(),s=a.map(e=>{let t=o.filter(t=>t.browserId===e.id);return{id:e.id,profileName:e.profileName,currentPageId:e.currentPageId,createdAt:e.createdAt.toISOString(),pages:t.map(e=>({id:e.id,url:e.url,title:e.title,createdAt:e.createdAt.toISOString()}))}}),c={totalBrowsers:a.length,totalPages:o.length};return t.json({browsers:s,stats:c})}catch(e){return console.error(`Failed to list browsers:`,e),t.json({error:`Failed to list browsers`,message:e instanceof Error?e.message:String(e)},500)}}),t.delete(`/browsers`,async t=>{try{return await e.get(r.BrowserService).closeAll(),t.json({success:!0,message:`All browsers closed`})}catch(e){return console.error(`Failed to close all browsers:`,e),t.json({error:`Failed to close all browsers`,message:e instanceof Error?e.message:String(e)},500)}}),t.delete(`/browsers/:id`,async t=>{try{let n=t.req.param(`id`),i=e.get(r.BrowserService);return i.getBrowser(n)?(await i.closeBrowser(n),t.json({success:!0,message:`Browser "${n}" closed`})):t.json({error:`Browser "${n}" not found`},404)}catch(e){return console.error(`Failed to close browser:`,e),t.json({error:`Failed to close browser`,message:e instanceof Error?e.message:String(e)},500)}}),t.delete(`/pages/:id`,async t=>{try{let n=t.req.param(`id`),i=e.get(r.PageRegistry).get(n);return i?i.page?(await i.page.close(),t.json({success:!0,message:`Page "${n}" closed`})):t.json({error:`Page "${n}" is in extension mode and cannot be closed via API`},400):t.json({error:`Page "${n}" not found`},404)}catch(e){return console.error(`Failed to close page:`,e),t.json({error:`Failed to close page`,message:e instanceof Error?e.message:String(e)},500)}}),t}function Mt(e){return e.toLocaleString(`en-US`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1})}function Nt(e){let t=Math.floor(e/1e3),n=Math.floor(t/60),r=Math.floor(n/60);return r>0?`${r}h ${n%60}m`:n>0?`${n}m ${t%60}s`:`${t}s`}function Pt(e){return Nt(new Date().getTime()-e.getTime())}const Ft=`table-container`,It=`stat-card`,Lt=`browser-row`,Rt=`page-row`,zt=`status-active`,Bt=`url-cell`,Z=`timestamp-cell`,Vt=`actions-cell`,Ht=`kill-btn`,Ut=`empty-state`;function Wt({browsers:e}){return e.length===0?R(`div`,{class:Ft,children:z(`div`,{class:Ut,children:[R(`h3`,{children:`No Active Browsers`}),R(`p`,{children:`Launch a browser using MCP tools to see it here.`})]})}):R(`div`,{class:Ft,children:z(`table`,{class:`browser-table`,children:[R(`thead`,{children:z(`tr`,{children:[R(`th`,{children:`ID`}),R(`th`,{children:`Status`}),R(`th`,{children:`URL / Title`}),R(`th`,{children:`Created`}),R(`th`,{children:`Age`}),R(`th`,{children:`Actions`})]})}),R(`tbody`,{id:`browser-table-body`,children:e.map(e=>z(_e,{children:[z(`tr`,{class:Lt,children:[R(`td`,{children:e.id}),R(`td`,{children:z(`span`,{class:zt,children:[e.pages.length,` page`,e.pages.length===1?``:`s`]})}),R(`td`,{children:e.profileName||`Default Profile`}),R(`td`,{class:Z,children:Mt(e.createdAt)}),R(`td`,{class:Z,children:Pt(e.createdAt)}),R(`td`,{class:Vt,children:R(`button`,{type:`button`,class:Ht,onclick:`dashboard.killBrowser('${e.id}')`,children:`Kill`})})]}),e.pages.map(t=>z(`tr`,{class:Rt,children:[z(`td`,{children:[t.id,e.currentPageId===t.id&&` (active)`]}),R(`td`,{children:R(`span`,{class:zt,children:`Open`})}),R(`td`,{class:Bt,title:t.url,children:t.title||t.url||`about:blank`}),R(`td`,{class:Z,children:Mt(t.createdAt)}),R(`td`,{class:Z,children:Pt(t.createdAt)}),R(`td`,{class:Vt,children:R(`button`,{type:`button`,class:Ht,onclick:`dashboard.killPage('${t.id}')`,children:`Close`})})]},t.id))]},e.id))})]})})}function Gt({stats:e}){return z(`div`,{class:`stats-container`,children:[z(`div`,{class:It,children:[R(`h3`,{children:`Active Browsers`}),R(`div`,{class:`value`,children:e.totalBrowsers})]}),z(`div`,{class:It,children:[R(`h3`,{children:`Active Pages`}),R(`div`,{class:`value`,children:e.totalPages})]})]})}function Kt({browsers:e,stats:t}){return z(`div`,{class:`dashboard-container`,children:[z(`div`,{class:`dashboard-header`,children:[z(`div`,{children:[R(`h1`,{children:`Playwright MCP Dashboard`}),R(`p`,{children:`Browser automation management (auto-refresh every 3 seconds)`})]}),z(`div`,{children:[R(`button`,{type:`button`,id:`refresh-btn`,class:`btn refresh-btn`,onclick:`dashboard.manualRefresh()`,children:`Refresh Now`}),R(`button`,{type:`button`,id:`kill-all-btn`,class:`btn btn-danger`,onclick:`dashboard.killAllBrowsers()`,children:`Kill All Browsers`})]})]}),R(Gt,{stats:t}),R(Wt,{browsers:e}),R(`script`,{children:ge(`
2
+ import{C as e,D as t,E as n,O as r,S as i,T as a,_ as o,a as s,b as c,c as l,d as u,f as d,g as f,h as p,i as m,l as h,m as g,n as _,o as v,p as y,s as b,t as ee,u as x,v as S,w as C,x as w,y as T}from"./streamable-http-vpqvkZ_0.mjs";import{stripTypeScriptTypes as E}from"node:module";import"reflect-metadata/lite";import{DEFAULT_PORT_RANGE as D,PortRegistryService as O}from"@agimon-ai/foundation-port-registry";import{createProcessLease as te,resolveSiblingRegistryPath as ne}from"@agimon-ai/foundation-process-registry";import{Container as k,ContainerModule as re}from"inversify";import A from"node:path";import{existsSync as j,readFileSync as ie,statSync as ae}from"node:fs";import{readFile as M,stat as oe}from"node:fs/promises";import{homedir as se}from"node:os";import{z as N}from"zod";import{SpanStatusCode as P}from"@opentelemetry/api";import{randomUUID as ce}from"node:crypto";import{coerceArgs as le,formatZodError as ue}from"@agimon-ai/foundation-validator";import{Server as de}from"@modelcontextprotocol/sdk/server/index.js";import{CallToolRequestSchema as fe,ListToolsRequestSchema as pe}from"@modelcontextprotocol/sdk/types.js";import{StdioServerTransport as me}from"@modelcontextprotocol/sdk/server/stdio.js";import{Command as F,Option as he}from"commander";import{serve as ge}from"@hono/node-server";import{Hono as I}from"hono";import{cors as _e}from"hono/cors";import{createNodeWebSocket as ve}from"@hono/node-ws";import{raw as ye}from"hono/html";import{Fragment as be}from"hono/jsx";import{jsx as L,jsxs as R}from"hono/jsx/jsx-runtime";var xe=`0.15.2`;const Se=process.env.BROWSE_TOOL_DEBUG_EXTENSION_RECORDING===`1`;function Ce(e,t){if(!Se)return;let n=t?` ${JSON.stringify(t)}`:``;console.log(`[ExtensionRecordingDebug] ${e}${n}`)}function we(e){let t=new I,n;try{n=e.get(r.TelemetryService)}catch{n=new f}return t.get(`/telemetry-config`,e=>{try{let t=new URL(e.req.url).origin;return e.json(o(process.env,t))}catch(t){return e.json({enabled:!1,error:t instanceof Error?t.message:String(t)},500)}}),t.get(`/tasks`,t=>{try{let n=e.get(r.ExtensionTaskQueue).getNextTask();return n?t.json({task:{id:n.id,tool:n.tool,arguments:n.arguments,telemetry:n.telemetry}}):t.json({})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/result`,async t=>{try{let n=await t.req.json();if(!n.taskId)return t.json({success:!1,error:`Missing taskId in request body`},400);let i=e.get(r.ExtensionTaskQueue),a=e.get(r.BrowserService),o={taskId:n.taskId,success:n.success,result:n.result,error:n.error},s=i.submitResult(o);return s?(s.browserId&&a.recordBrowserActivity(s.browserId,s.pageId),t.json({success:!0})):t.json({success:!1,error:`Task ${n.taskId} not found or already completed`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/status`,t=>{try{let n=e.get(r.ExtensionTaskQueue),i=n.getConnectionStatus(),a={connected:i.connected,lastPollAt:i.lastPollAt?.toISOString(),lastResultAt:i.lastResultAt?.toISOString(),pendingTasks:i.pendingTasks,queueSize:n.queueSize};return t.json(a)}catch(e){return t.json({connected:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/register`,async t=>{try{let n=await t.req.json();if(!n.browserId)return t.json({success:!1,error:`Missing browserId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).register(n);return t.json({success:!0,session:{id:i.id,browserId:i.browserId,controlMode:i.controlMode,createdAt:i.createdAt.toISOString()}})}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/heartbeat`,async t=>{try{let n=await t.req.json();if(!n.sessionId)return t.json({success:!1,error:`Missing sessionId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).heartbeat(n);return i?t.json({success:!0,session:{id:i.id,controlMode:i.controlMode,handoffRequested:i.handoffRequested,lastHeartbeatAt:i.lastHeartbeatAt.toISOString()}}):t.json({success:!1,error:`Session ${n.sessionId} not found`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/tab-mapped`,async t=>{try{let n=await t.req.json();if(!n.pageId||typeof n.tabId!=`number`)return t.json({success:!1,error:`Missing pageId or tabId in request body`},400);let i=e.get(r.PageRegistry),a=e.get(r.BrowserService),o=i.get(n.pageId);return o?(o.extensionTabId=n.tabId,a.recordBrowserActivity(o.browserId,n.pageId),t.json({success:!0})):t.json({success:!1,error:`Page ${n.pageId} not found`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/recording`,async t=>{try{let n=await t.req.json();if(!n.browserId)return t.json({success:!1,error:`Missing browserId in request body`},400);let i=e.get(r.BrowserService),a=await i.persistExtensionRecordingArtifact(n.browserId,n.videoBase64);return Ce(`artifact received`,{browserId:n.browserId,videoBase64Size:n.videoBase64?.length??0,persisted:a}),a?(i.recordBrowserActivity(n.browserId),t.json({success:!0})):t.json({success:!1,error:`Browser "${n.browserId}" has no active recording target`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/recording/chunk`,async t=>{try{let i=await t.req.json();if(!i.browserId||!i.chunkBase64)return t.json({success:!1,error:`Missing browserId or chunkBase64 in request body`},400);let a=e.get(r.BrowserService),o=await a.persistExtensionRecordingChunk(i.browserId,i.chunkBase64);return Ce(`chunk received`,{browserId:i.browserId,chunkIndex:i.chunkIndex,mimeType:i.mimeType,persisted:!!o,chunkBase64Size:i.chunkBase64.length}),o?(n.log(`debug`,`extension recording chunk received`,{attributes:{"browse_tool.extension.recording.chunk_received":!0,"browse_tool.browser.id":i.browserId,"browse_tool.recording.chunk_bytes":o.chunkBytes,"browse_tool.recording.total_bytes":o.totalBytes,"browse_tool.recording.chunk_count":o.chunkCount,...typeof i.chunkIndex==`number`?{"browse_tool.recording.chunk_index":i.chunkIndex}:{},...typeof i.mimeType==`string`?{"browse_tool.recording.mime_type":i.mimeType}:{}}}),a.recordBrowserActivity(i.browserId),t.json({success:!0})):t.json({success:!1,error:`Browser "${i.browserId}" has no active recording target`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/browser-log`,async e=>{try{let t=await e.req.json();return!t.message||typeof t.message!=`string`?e.json({success:!1,error:`Missing message in request body`},400):(n.log(t.level??`info`,t.message,{attributes:{"browse_tool.extension.log_relay":!0,...typeof t.attributes==`object`&&t.attributes!==null?t.attributes:{}}}),Ce(`browser log relayed`,{level:t.level??`info`,message:t.message,attributes:t.attributes}),e.json({success:!0}))}catch(t){return e.json({success:!1,error:t instanceof Error?t.message:String(t)},500)}}),t.post(`/handoff`,async t=>{try{let n=await t.req.json();if(!n.sessionId)return t.json({success:!1,error:`Missing sessionId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).requestHandoff(n);return i?t.json({success:!0,message:`Handoff requested. AI will take control when ready.`,session:{id:i.id,controlMode:i.controlMode,handoffRequested:i.handoffRequested}}):t.json({success:!1,error:`Session ${n.sessionId} not found`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.post(`/handoff/acknowledge`,async t=>{try{let n=await t.req.json();if(!n.sessionId)return t.json({success:!1,error:`Missing sessionId in request body`},400);let i=e.get(r.ExtensionSessionRegistry).acknowledgeHandoff(n.sessionId);return i?t.json({success:!0,message:`Handoff acknowledged. AI now has control.`,session:{id:i.id,controlMode:i.controlMode,handoffRequested:i.handoffRequested}}):t.json({success:!1,error:`Session ${n.sessionId} not found or no handoff pending`},404)}catch(e){return t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/sessions`,t=>{try{let n=e.get(r.ExtensionSessionRegistry).listSessions();return t.json({sessions:n.map(e=>({id:e.id,browserId:e.browserId,tabId:e.tabId,currentUrl:e.currentUrl,controlMode:e.controlMode,activeSpecPath:e.activeSpecPath,handoffRequested:e.handoffRequested,createdAt:e.createdAt.toISOString(),lastHeartbeatAt:e.lastHeartbeatAt.toISOString()}))})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t}function Te(e){if(e==null||e===``)return;let t=typeof e==`number`?e:typeof e==`string`?Number.parseInt(e,10):NaN;if(!Number.isInteger(t)||t<=0||t>65535)throw Error(`Invalid port: ${e}`);return t}const Ee=[`pnpm-workspace.yaml`,`nx.json`,`.git`],De=`browse-tool-chrome`,Oe=`tool`,ke=process.env.NODE_ENV||`development`,Ae=`127.0.0.1`;function je(e=process.cwd()){let t=A.resolve(e);for(;;){for(let e of Ee)if(j(A.join(t,e)))return t;let e=A.dirname(t);if(e===t)return process.cwd();t=e}}function Me(){return new O(process.env.PORT_REGISTRY_PATH)}async function Ne(e,t,n){let r=Me(),i=await r.reservePort({repositoryPath:e,serviceName:De,serviceType:Oe,environment:ke,preferredPort:t,portRange:n,pid:process.pid,host:Ae,force:!0,metadata:{transport:`stdio`,mode:`chrome-serve`}});if(!i.success||!i.record)throw Error(i.error||`Failed to reserve port ${t}`);let a=!1;return{port:i.record.port,release:async()=>{if(a)return;a=!0;let n=await r.releasePort({repositoryPath:e,serviceName:De,serviceType:Oe,environment:ke,pid:process.pid});if(!n.success&&!n.error?.includes(`No matching registry entry`))throw Error(n.error||`Failed to release port ${t}`)}}}function Pe(){return new re(e=>{e.bind(r.ExtensionTaskQueue).to(p).inSingletonScope(),e.bind(r.ExtensionToolDelegator).to(g).inSingletonScope()})}function Fe(e){let t=new de({name:`browse-tool-chrome`,version:`0.1.0`},{capabilities:{tools:{}}}),n=e.getSupportedTools().map(e=>({name:e,description:`Browser automation tool (Chrome extension mode): ${e}`,inputSchema:{type:`object`,properties:{},additionalProperties:!0}}));return t.setRequestHandler(pe,async()=>({tools:n})),t.setRequestHandler(fe,async t=>{let{name:n,arguments:r}=t.params;return await e.executeTool(n,r||{})}),t}const Ie=new F(`chrome-serve`).description(`[DEPRECATED] Start MCP server with Chrome extension HTTP polling for bot-detection-free browser automation`).option(`-p, --port <port>`,`HTTP server port for extension polling`).option(`-v, --verbose`,`Enable verbose output`,!1).option(`--wait-for-extension`,`Wait for extension to connect before accepting MCP requests`,!1).action(async e=>{console.error(``),console.error(`╔════════════════════════════════════════════════════════════════╗`),console.error(`║ DEPRECATED: chrome-serve is deprecated. ║`),console.error(`║ Use mcp-serve instead - extension routes are now ║`),console.error(`║ automatically available in the HTTP server at /extension/*. ║`),console.error(`╚════════════════════════════════════════════════════════════════╝`),console.error(``);let t,n;try{let i=Te(e.port),a=je(process.cwd()),o=process.env.PORT_REGISTRY_PATH;o&&(process.env.PROCESS_REGISTRY_PATH=ne(o,`processes.json`)),t=await Ne(a,i??D.min,i?{min:i,max:i}:D);let s=t.port;n=await te({repositoryPath:a,serviceName:De,serviceType:Oe,environment:ke,pid:process.pid,port:s,host:Ae,command:process.argv[1],args:process.argv.slice(2),metadata:{transport:`stdio`,command:`chrome-serve`,waitForExtension:e.waitForExtension}}),e.verbose&&(console.error(`Chrome Extension MCP Server starting...`),console.error(` HTTP Port: ${s}`),console.error(` Wait for extension: ${e.waitForExtension}`));let c=new k({defaultScope:`Singleton`});c.load(Pe());let l=c.get(r.ExtensionTaskQueue),u=c.get(r.ExtensionToolDelegator),d=new I;d.use(`*`,_e({origin:e=>e??null,credentials:!0,allowMethods:[`GET`,`POST`,`OPTIONS`],allowHeaders:[`Content-Type`,`Accept`]}));let f=we(c);d.route(`/extension`,f),d.get(`/health`,e=>{let t=l.getConnectionStatus();return e.json({status:`healthy`,service:`browse-tool-chrome`,extension:t})});let p=ge({fetch:d.fetch,port:s});p.on(`error`,e=>{e.code===`EADDRINUSE`?(console.error(`Error [PORT_IN_USE]: Port ${s} is already in use.`),console.error(`Recovery: Try a different port with --port <port>`)):console.error(`Error [SERVER_ERROR]: HTTP server error: ${e.message}`),process.exit(1)}),console.error(`HTTP server started on port ${s}`),console.error(`Waiting for Chrome extension to connect...`),console.error(`Extension should poll: http://localhost:${s}/extension/tasks`),e.waitForExtension&&(console.error(`Waiting for extension connection before starting MCP...`),await new Promise(e=>{let t=setInterval(()=>{l.getConnectionStatus().connected&&(clearInterval(t),console.error(`Extension connected!`),e())},500)}));let m=Fe(u),h=new me;await m.connect(h),console.error(`Chrome extension MCP server started on stdio`);let g=async e=>{console.error(`\nReceived ${e}, shutting down gracefully...`);try{l.clearAllTasks(`Server shutting down`),await h.close(),p.close(),await n.release({kill:!1}),await t.release(),process.exit(0)}catch(e){console.error(`Error during shutdown:`,e),process.exit(1)}};process.on(`SIGINT`,()=>g(`SIGINT`)),process.on(`SIGTERM`,()=>g(`SIGTERM`))}catch(e){if(n||t)try{await n?.release(),await t?.release()}catch{}let r=e instanceof Error?e.message:String(e);console.error(`Error [SERVER_ERROR]: Failed to start Chrome extension MCP server: ${r}`),console.error(`Recovery: Check that the port is available and try again.`),process.exit(1)}}),Le=`BROWSE_TOOL_CONFIG`,Re=`.browse-tool`,ze=`config.json`,Be=[`json`,`text`,`quiet`],Ve={commands:{},tools:{}},He=N.record(N.string(),N.unknown()),Ue=N.object({mcpServe:N.object({type:N.string().optional(),browser:N.string().optional(),headless:N.boolean().optional(),profile:N.string().optional(),mode:N.string().optional(),host:N.string().optional(),port:N.coerce.number().int().positive().optional(),httpPort:N.coerce.number().int().positive().optional(),tags:N.string().optional(),exclude:N.string().optional(),customTools:N.string().optional(),chromeForTestingPath:N.string().optional(),snippetsDir:N.string().optional(),registryPath:N.string().optional(),registryDir:N.string().optional(),pidsDir:N.string().optional(),profilesDir:N.string().optional(),proxyConfigDir:N.string().optional()}).partial().optional(),httpServe:N.object({port:N.coerce.number().int().positive().optional(),headless:N.boolean().optional(),idleTimeout:N.coerce.number().positive().optional(),host:N.string().optional(),registryDir:N.string().optional(),registryPath:N.string().optional(),pidsDir:N.string().optional(),profilesDir:N.string().optional(),snippetsDir:N.string().optional(),proxyConfigDir:N.string().optional()}).partial().optional(),exec:N.object({format:N.enum(Be).optional(),color:N.boolean().optional(),port:N.coerce.number().int().positive().optional()}).partial().optional(),tools:N.object({format:N.enum(Be).optional(),color:N.boolean().optional(),port:N.coerce.number().int().positive().optional()}).partial().optional(),status:N.record(N.string(),N.unknown()).optional(),stop:N.record(N.string(),N.unknown()).optional()}),We=N.object({commands:Ue.default({}),tools:N.record(N.string(),He).default({})});let Ge={config:Ve};function Ke(e){for(let t=0;t<e.length;t+=1){let n=e[t];if(n===`--config`)return e[t+1];if(n.startsWith(`--config=`))return n.slice(9)}}function qe(e){let t=A.resolve(e);if(!j(t))throw Error(`Config path not found: ${t}`);return ae(t).isDirectory()?A.join(t,ze):t}function Je(e,t){let n=t.env??process.env,r=t.cwd??process.cwd(),i=t.homeDir??se(),a=Ke(e);if(a)return qe(a);if(n[Le])return qe(n[Le]);let o=A.join(r,Re,ze);if(j(o))return o;let s=A.join(i,Re,ze);if(j(s))return s}function Ye(e){let t=ie(e,`utf8`),n=JSON.parse(t);return We.parse(n)}function Xe(e,t={}){let n=Je(e,t);return n?{configPath:n,config:Ye(n)}:{config:Ve}}function Ze(e,t={}){return Ge=Xe(e,t),Ge}function z(e){return(Ge.config.commands??{})[e]??{}}function Qe(e){return Ge.config.tools?.[e]??{}}function B(e,t,n,r,i){let a=e.getOptionValueSourceWithGlobals(t);return a!==void 0&&a!==`default`&&a!==`implied`?n:i===void 0?r===void 0?n:r:i}var $e=class{port;exactPort;timeout;serverPort=null;constructor(e={}){this.port=e.port??u,this.exactPort=e.exactPort??!1,this.timeout=e.timeout??6e4}async ensureServer(){if(this.serverPort!==null)return this.serverPort;let e=await v().get(r.HttpServerManager).ensureRunning(this.port,{exactPort:this.exactPort});if(!e.running||!e.port)throw Error(e.error??`Failed to start HTTP server. Try running "browse-tool http-serve" manually.`);return this.serverPort=e.port,e.port}async getBaseUrl(){return`http://localhost:${await this.ensureServer()}`}async listTools(){let e=await this.getBaseUrl(),t=new AbortController,n=setTimeout(()=>t.abort(),this.timeout);try{let n=await fetch(`${e}/tools`,{method:`GET`,headers:{Accept:`application/json`},signal:t.signal});if(!n.ok)throw Error(`HTTP ${n.status}: ${n.statusText}`);let r=await n.json();if(r.error)throw Error(r.error);return r.tools}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(n)}}async listCustomTools(e){let t=await this.getBaseUrl(),n=new AbortController,r=setTimeout(()=>n.abort(),this.timeout);try{let r=new URL(`/custom-tools`,t);r.searchParams.set(`dir`,e);let i=await fetch(r,{method:`GET`,headers:{Accept:`application/json`},signal:n.signal});if(!i.ok)throw Error(`HTTP ${i.status}: ${i.statusText}`);let a=await i.json();if(a.error)throw Error(a.error);return a.tools}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(r)}}async listBrowsers(){let e=await this.getBaseUrl(),t=new AbortController,n=setTimeout(()=>t.abort(),this.timeout);try{let n=await fetch(`${e}/browsers`,{method:`GET`,headers:{Accept:`application/json`},signal:t.signal});if(!n.ok)throw Error(`HTTP ${n.status}: ${n.statusText}`);let r=await n.json();if(r.error)throw Error(r.error);return r.browsers}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(n)}}async execute(e,t){let n=await this.getBaseUrl(),r=e===`run_spec`?void 0:this.timeout,i=new AbortController,a=r===void 0?void 0:setTimeout(()=>i.abort(),r);try{let a=await fetch(`${n}/execute`,{method:`POST`,headers:{"Content-Type":`application/json`,Accept:`application/json`},body:JSON.stringify({tool:e,arguments:t}),signal:r===void 0?void 0:i.signal});if(!a.ok){let e=await a.text();throw Error(`HTTP ${a.status}: ${e||a.statusText}`)}let o=await a.json();return o.success?o.result??{content:[{type:`text`,text:`No result returned`}]}:{content:[{type:`text`,text:o.error??`Unknown error`}],isError:!0}}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{a&&clearTimeout(a)}}async executeCustomTool(e,t,n){let r=await this.getBaseUrl(),i=new AbortController,a=setTimeout(()=>i.abort(),this.timeout);try{let a=new URL(`/custom-tools`,r);a.searchParams.set(`dir`,e);let o=await fetch(a,{method:`POST`,headers:{"Content-Type":`application/json`,Accept:`application/json`},body:JSON.stringify({tool:t,arguments:n}),signal:i.signal});if(!o.ok){let e=await o.text();throw Error(`HTTP ${o.status}: ${e||o.statusText}`)}let s=await o.json();return s.success?s.result??{content:[{type:`text`,text:`No result returned`}]}:{content:[{type:`text`,text:s.error??`Unknown error`}],isError:!0}}catch(e){throw e instanceof Error&&e.name===`AbortError`?Error(`Request timeout after ${this.timeout}ms`,{cause:e}):this.wrapConnectionError(e)}finally{clearTimeout(a)}}wrapConnectionError(e){return e instanceof Error?e.message.includes(`ECONNREFUSED`)||e.message.includes(`fetch failed`)?Error(`Cannot connect to browse-tool HTTP server.\n\nTry one of the following:\n 1. Start the server: browse-tool http-serve\n 2. Check if another process is using port ${this.port}\n 3. Run with a different port: browse-tool --port 3201 <command>\n\nOriginal error: ${e.message}`,{cause:e}):e:Error(String(e),{cause:e})}};function V(e){return new $e(e)}const et={reset:`\x1B[0m`,red:`\x1B[31m`,green:`\x1B[32m`,yellow:`\x1B[33m`,blue:`\x1B[34m`,magenta:`\x1B[35m`,cyan:`\x1B[36m`,gray:`\x1B[90m`,bold:`\x1B[1m`};function H(e,t,n){return n?`${et[t]}${e}${et.reset}`:e}function tt(e,t){let{format:n,color:r}=t;return n===`quiet`?``:n===`json`?nt(e):rt(e,r)}function nt(e){return JSON.stringify(e,null,2)}function rt(e,t){let n=[];e.isError&&n.push(H(`Error:`,`red`,t));for(let r of e.content)r.type===`text`?n.push(it(r,t)):r.type===`image`?n.push(st(r,t)):n.push(H(`[Unknown content type: ${r.type}]`,`yellow`,t));return n.join(`
3
+ `)}function it(e,t){let n=e.text;try{return at(JSON.parse(n),t)}catch{return n}}function at(e,t){if(typeof e!=`object`||!e)return String(e);let n=[];for(let[r,i]of Object.entries(e)){let e=H(r,`cyan`,t),a=ot(i,t);n.push(`${e}: ${a}`)}return n.join(`
4
+ `)}function ot(e,t){return e===null?H(`null`,`gray`,t):e===void 0?H(`undefined`,`gray`,t):typeof e==`boolean`?H(String(e),e?`green`:`red`,t):typeof e==`number`?H(String(e),`yellow`,t):typeof e==`string`?e.startsWith(`http://`)||e.startsWith(`https://`)?H(e,`blue`,t):e:Array.isArray(e)?e.length===0?H(`[]`,`gray`,t):JSON.stringify(e,null,2):typeof e==`object`?JSON.stringify(e,null,2):String(e)}function st(e,t){let{mimeType:n,data:r}=e;return H(`[Image: ${n}, ~${Math.round(r.length*3/4/1024)}KB base64 data]`,`magenta`,t)}function U(e,t){return H(`Error: ${e instanceof Error?e.message:e}`,`red`,t)}function ct(e,t){let n=Math.max(...e.map(e=>e.name.length)),r=[];for(let i of e){let e=H(i.name.padEnd(n),`cyan`,t);r.push(` ${e} ${i.description}`)}return r.join(`
5
+ `)}function lt(e,t){let n=z(`tools`),r=B(e,`format`,t.format,n.format),i=B(e,`color`,t.color,n.color);return{port:B(e,`port`,t.port,n.port===void 0?void 0:String(n.port),process.env.PLAYWRIGHT_PORT),formatterOptions:{format:r,color:i}}}const ut=new F(`list-custom-tools`).description(`List custom tools from a tools directory`).argument(`<dir>`,`Path to the tools directory containing tools.yaml`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u)).action(async function(e,t){let{port:n,formatterOptions:r}=lt(this,t);try{let t=await V({port:Number.parseInt(n,10)}).listCustomTools(e),i=r.format===`text`?ct(t,r.color):r.format===`quiet`?``:JSON.stringify(t,null,2);i&&console.log(i)}catch(e){console.error(U(e instanceof Error?e:String(e),r.color)),process.exit(1)}}),dt=new F(`exec-custom-tool`).description(`Execute a custom tool from a tools directory`).argument(`<dir>`,`Path to the tools directory containing tools.yaml`).argument(`<tool>`,`Custom tool name to execute`).argument(`[args]`,`JSON arguments for the tool`,`{}`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u)).action(async function(e,t,n,r){let{port:i,formatterOptions:a}=lt(this,r);try{let r;try{r=JSON.parse(n)}catch{console.error(U(`Invalid JSON arguments: ${n}`,a.color)),process.exit(1);return}let o=await V({port:Number.parseInt(i,10)}).executeCustomTool(e,t,r),s=tt(o,a);s&&console.log(s),o.isError&&process.exit(1)}catch(e){console.error(U(e instanceof Error?e:String(e),a.color)),process.exit(1)}});new F(`custom-tools`).description(`List and execute custom tools through the HTTP server`).addCommand(ut).addCommand(dt);const ft=new F(`docker-build-cft`).description(`Build the Chrome for Testing Docker image used by vm mode`).option(`--cft-version <version>`,`Chrome for Testing version to build`,t).option(`--image <image>`,`Docker image tag to produce`,a).option(`--platform <platform>`,`Docker target platform`,n).action(async t=>{try{let n=await i({version:t.cftVersion,image:t.image,platform:t.platform,stdio:`inherit`});console.log(`Built Docker image ${n.image}`),console.log(` Version: ${n.version}`),console.log(` Platform: ${n.platform}`),console.log(` Archive: ${e(n.platform)}`)}catch(e){console.error(e instanceof Error?e.message:String(e)),process.exit(1)}}),pt=new F(`exec`).description(`Execute a tool directly with JSON arguments`).argument(`<tool>`,`Tool name to execute (e.g., browser_launch)`).argument(`[args]`,`JSON arguments for the tool`,`{}`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u)).action(async function(e,t,n){let r=z(`exec`),i=B(this,`format`,n.format,r.format),a=B(this,`color`,n.color,r.color),o=B(this,`port`,n.port,r.port===void 0?void 0:String(r.port),process.env.PLAYWRIGHT_PORT),s=this.getOptionValueSourceWithGlobals(`port`),c=s!==void 0&&s!==`default`&&s!==`implied`,l={format:i,color:a};try{let n;try{n=JSON.parse(t)}catch{console.error(U(`Invalid JSON arguments: ${t}`,l.color)),process.exit(1)}let r=await V({port:Number.parseInt(o,10),exactPort:c}).execute(e,n),i=tt(r,l);i&&console.log(i),r.isError&&process.exit(1)}catch(e){console.error(U(e instanceof Error?e:String(e),l.color)),process.exit(1)}}),W=`pageId`,G=`browserId`,mt=N.record(N.string(),N.unknown()),ht=N.object({type:N.literal(`object`),properties:N.record(N.string(),N.unknown()).optional(),required:N.array(N.string()).optional(),additionalProperties:N.boolean().optional()}).passthrough(),gt=N.object({name:N.string().min(1),description:N.string().min(1),script:N.string().min(1),suggestionActions:N.string().min(1).optional(),capabilities:mt,inputSchema:ht}),_t=N.object({tools:N.array(gt)}),vt=process.env.BROWSE_TOOL_DEBUG_CUSTOM_TOOLS===`1`;function yt(e){return JSON.stringify(e,(e,t)=>typeof t==`string`&&t.length>240?`${t.slice(0,240)}...<trimmed>`:t)}function bt(e,t){if(vt){if(t){console.error(`[CustomToolService] ${e}`,t);return}console.error(`[CustomToolService] ${e}`)}}function K(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}function xt(e){return K(e)&&Array.isArray(e.content)}function q(e){let t=e.trim();return t===``?``:t.startsWith(`"`)&&t.endsWith(`"`)||t.startsWith(`[`)&&t.endsWith(`]`)||t.startsWith(`{`)&&t.endsWith(`}`)?JSON.parse(t):t.startsWith(`'`)&&t.endsWith(`'`)?t.slice(1,-1):t===`true`?!0:t===`false`?!1:t===`null`?null:/^-?\d+(\.\d+)?$/.test(t)?Number(t):t}function St(e){if(!e||Object.keys(e).length===0)return;let t={};for(let[n,r]of Object.entries(e))if(r!=null){if(typeof r==`string`||typeof r==`number`||typeof r==`boolean`){t[n]=r;continue}t[n]=JSON.stringify(r)}return Object.keys(t).length>0?t:void 0}function Ct(e){return e.replace(/^\uFEFF/,``).split(/\r?\n/).map(e=>{if(e.includes(` `))throw Error(`Tab indentation is not supported in tools.yaml`);let t=e.replace(/\s+#.*$/,``);return t.trim().length===0?null:{indent:t.match(/^ */)?.[0].length??0,text:t.trim()}}).filter(e=>e!==null)}function J(e,t,n){let r=e[t];if(!r||r.indent!==n)throw Error(`Invalid indentation in tools.yaml at line ${t+1}`);return r.text.startsWith(`-`)?wt(e,t,n):Tt(e,t,n)}function wt(e,t,n){let r=[],i=t;for(;i<e.length;){let t=e[i];if(t.indent<n||t.indent!==n||!t.text.startsWith(`-`))break;let a=t.text.slice(1).trim();if(a===``){let t=e[i+1];if(!t||t.indent<=n){r.push(null),i+=1;continue}let[a,o]=J(e,i+1,t.indent);r.push(a),i=o;continue}if(a.includes(`:`)){let[t,o]=Et(a),s={};if(o===void 0){let r=e[i+1];if(!r||r.indent<=n)throw Error(`Expected nested value for "${t}" in tools.yaml`);let[a,o]=J(e,i+1,r.indent);s[t]=a,i=o}else s[t]=q(o),i+=1;for(;i<e.length&&e[i].indent>n;){let t=e[i];if(t.indent!==n+2||t.text.startsWith(`-`)){let[n,r]=J(e,i,t.indent);if(!K(n))throw Error(`Expected object entry in tools.yaml at line ${i+1}`);Object.assign(s,n),i=r;continue}let[r,a]=Et(t.text);if(a===void 0){let n=e[i+1];if(!n||n.indent<=t.indent)throw Error(`Expected nested value for "${r}" in tools.yaml`);let[a,o]=J(e,i+1,n.indent);s[r]=a,i=o;continue}s[r]=q(a),i+=1}r.push(s);continue}r.push(q(a)),i+=1}return[r,i]}function Tt(e,t,n){let r={},i=t;for(;i<e.length;){let t=e[i];if(t.indent<n||t.indent!==n||t.text.startsWith(`-`))break;let[a,o]=Et(t.text);if(o===void 0){let t=e[i+1];if(!t||t.indent<=n){r[a]=null,i+=1;continue}let[o,s]=J(e,i+1,t.indent);r[a]=o,i=s;continue}r[a]=q(o),i+=1}return[r,i]}function Et(e){let t=e.indexOf(`:`);if(t===-1)throw Error(`Invalid tools.yaml entry: "${e}"`);let n=e.slice(0,t).trim(),r=e.slice(t+1).trim();return[n,r===``?void 0:r]}function Dt(e){let t=Ct(e);if(t.length===0)return{};let[n]=J(t,0,t[0].indent);return n}function Ot(e){e.inputSchema.properties===void 0&&(e.inputSchema.properties={});let t=e.inputSchema.properties;if(!K(t))throw Error(`Custom tool "${e.name}" inputSchema.properties must be an object`);let n=t[W],r=t[G];if(n!==void 0&&!(K(n)&&n.type===`string`))throw Error(`Custom tool "${e.name}" must define inputSchema.properties.pageId as type "string"`);if(r!==void 0&&!(K(r)&&r.type===`string`))throw Error(`Custom tool "${e.name}" must define inputSchema.properties.browserId as type "string"`);n===void 0&&(t[W]={type:`string`,description:`Browse-tool page ID to run this custom tool against. Either pageId or browserId is required at call time.`}),r===void 0&&(t[G]={type:`string`,description:`Browse-tool browser ID to run this custom tool against when no pageId is supplied. The tool will use the browser's current page or open a new one.`})}async function kt(e){let t=E(await M(e,`utf8`),{mode:`strip`});return await import(`data:text/javascript;base64,${Buffer.from(t,`utf8`).toString(`base64`)}`)}function At(e,t){let n=typeof t.run==`function`?t.run:void 0;if(!n)throw Error(`Custom tool script "${e}" must export a "run" function`);return{execute:n}}function jt(e,t){return typeof e.suggestionActions==`string`&&e.suggestionActions.trim().length>0?e:{...e,suggestionActions:t}}function Mt(e,t){let n=e.content[0];if(n?.type===`text`&&typeof n.text==`string`)try{let r=JSON.parse(n.text);if(K(r)){let i=jt(r,t);return{...e,content:[{...n,text:JSON.stringify(i,null,2)},...e.content.slice(1)]}}}catch{}return{...e,content:[...e.content,{type:`text`,text:`suggestionActions: ${t}`}]}}function Nt(e,t,n){if(xt(t))return n?Mt(t,n):t;if(typeof t==`string`)return n?{content:[{type:`text`,text:JSON.stringify({result:t,suggestionActions:n},null,2)}]}:{content:[{type:`text`,text:t}]};if(t===void 0)return n?{content:[{type:`text`,text:JSON.stringify({result:`Custom tool "${e}" completed successfully`,suggestionActions:n},null,2)}]}:{content:[{type:`text`,text:`Custom tool "${e}" completed successfully`}]};let r=K(t)&&n?jt(t,n):t;return{content:[{type:`text`,text:JSON.stringify(r,null,2)}]}}var Pt=class{constructor(e,t,n=new f,r){this.pageRegistry=e,this.extensionTaskQueue=t,this.telemetry=n,this.browserService=r}resolveToolPage(e,t,n){if(n.page)return n.page;if(n.mode===`extension`&&this.extensionTaskQueue){let e=new S(this.extensionTaskQueue);return e.setTarget(t,n.browserId),e}throw Error(`Custom tool "${e}" requires a supported page context`)}toPageSummary(e,t){return{pageId:t.id,url:t.url,title:t.title,active:e===t.id}}async createPageForBrowser(e,t,n){if(!this.browserService)throw Error(`Custom tool "${e}" requires browser service support to create a page`);let r=this.browserService.getBrowser(t);if(!r)throw Error(`Browser "${t}" not found`);let i=n?.setAsCurrent!==!1;if(r.mode===`extension`||r.mode===`vm`){if(!this.extensionTaskQueue)throw Error(`Custom tool "${e}" requires extension task support to create a page`);let a=this.pageRegistry.registerExtensionPage(t,void 0,n?.url,!1);try{let o=await this.extensionTaskQueue.queueTask(`browser_new_page`,{browserId:t,pageId:a,url:n?.url,setAsCurrent:i},1e4,t);if(!o.success)throw Error(o.error??`Custom tool "${e}" failed to create a page`);let s=o.result?.content[0]?.type===`text`?o.result.content[0].text:void 0,c={};if(typeof s==`string`&&s.length>0)try{c=JSON.parse(s)}catch{c={}}let l=this.pageRegistry.get(a);if(!l)throw Error(`Page "${a}" was not registered`);l.url=c.url??l.url,l.title=c.title??l.title,l.extensionTabId=c.tabId??l.extensionTabId;let u=await this.waitForResolvedPageMetadata(a);return r.pageIds.add(a),(i||!r.currentPageId)&&this.browserService.setCurrentPage(t,a),this.browserService.recordBrowserActivity(t,a),{...this.toPageSummary(r.currentPageId,u),page:this.resolveToolPage(e,a,u)}}catch(e){throw this.pageRegistry.remove(a),e}}let{pageId:a,page:o}=await this.browserService.newPage(t);n?.url&&(await o.goto(n.url),await this.pageRegistry.updateMetadata(a)),i&&this.browserService.setCurrentPage(t,a),this.browserService.recordBrowserActivity(t,a);let s=this.pageRegistry.get(a);if(!s)throw Error(`Page "${a}" was not registered`);return{...this.toPageSummary(r.currentPageId,s),page:this.resolveToolPage(e,a,s)}}async waitForResolvedPageMetadata(e){let t=Date.now(),n=this.pageRegistry.get(e);for(;n&&Date.now()-t<1500;){let t=typeof n.url==`string`&&n.url.length>0,r=typeof n.title==`string`&&n.title.length>0&&n.title!==`Extension Tab`;if(t&&r)return n;await new Promise(e=>setTimeout(e,50)),n=this.pageRegistry.get(e)}if(!n)throw Error(`Page "${e}" was not registered`);return n}async resolveExecutionContext(e,t){let n=typeof t[W]==`string`&&t[W].length>0?t[W]:void 0,r=typeof t[G]==`string`&&t[G].length>0?t[G]:void 0;if(!n&&!r)throw Error(`Custom tool "${e}" requires a string pageId or browserId`);if(n){let t=this.pageRegistry.get(n);if(!t)throw Error(`Page "${n}" not found`);let i=c();if(i&&t.ownerId&&t.ownerId!==i)throw Error(`Custom tool "${e}" cannot use page "${n}" owned by another session`);if(r&&t.browserId!==r)throw Error(`Custom tool "${e}" received pageId "${n}" for browser "${t.browserId}", not "${r}"`);let a=this.createBrowserHelper(e,t.browserId);return{pageId:n,pageEntry:t,page:this.resolveToolPage(e,n,t),browser:a}}let i=this.createBrowserHelper(e,r);try{let e=await i.getCurrentPage(),t=this.pageRegistry.get(e.pageId);if(!t)throw Error(`Page "${e.pageId}" not found`);return{pageId:e.pageId,pageEntry:t,page:e.page,browser:i}}catch(e){if(!(e instanceof Error?e.message:String(e)).includes(`has no pages`))throw e;let t=await i.newPage(),n=this.pageRegistry.get(t.pageId);if(!n)throw Error(`Page "${t.pageId}" not found`,{cause:e});return{pageId:t.pageId,pageEntry:n,page:t.page,browser:i}}}createBrowserHelper(e,t){return{browserId:t,mode:this.browserService?.getBrowser(t)?.mode??`extension`,listPages:async()=>{let e=this.browserService?.getBrowser(t);if(!e)throw Error(`Browser "${t}" not found`);return this.pageRegistry.findByBrowser(t).map(t=>this.toPageSummary(e.currentPageId,t))},getPage:async n=>{let r=this.pageRegistry.get(n);if(!r||r.browserId!==t)throw Error(`Page "${n}" not found in browser "${t}"`);let i=this.browserService?.getBrowser(t);return{...this.toPageSummary(i?.currentPageId??null,r),page:this.resolveToolPage(e,n,r)}},getCurrentPage:async()=>{let n=this.browserService?.getBrowser(t);if(!n)throw Error(`Browser "${t}" not found`);let r=c(),i;if(r){let e=this.pageRegistry.findByOwner(r).filter(e=>e.browserId===t);if(i=e[e.length-1]?.id,!i)throw Error(`Browser "${t}" has no pages owned by this session`)}else i=n.currentPageId??this.pageRegistry.findByBrowser(t)[0]?.id??void 0;if(!i)throw Error(`Browser "${t}" has no pages`);let a=this.pageRegistry.get(i);if(!a)throw Error(`Page "${i}" not found`);return{...this.toPageSummary(n.currentPageId,a),page:this.resolveToolPage(e,i,a)}},newPage:async n=>this.createPageForBrowser(e,t,n)}}createToolLogger(e,t,n){let r={"browse_tool.tool.name":e,"browse_tool.page.id":t,"browse_tool.browser.id":n.browserId,"browse_tool.execution.mode":n.mode},i=(e,t,n)=>{this.telemetry.log(e,t,{attributes:St({...r,...n?.attributes??{}}),exception:n?.exception})};return{getTraceContext:()=>this.telemetry.getActiveTraceContext(),trace:(e,t)=>i(`trace`,e,t),debug:(e,t)=>i(`debug`,e,t),info:(e,t)=>i(`info`,e,t),warn:(e,t)=>i(`warn`,e,t),error:(e,t)=>i(`error`,e,t),fatal:(e,t)=>i(`fatal`,e,t)}}async listTools(e){return(await this.loadTools(e)).map(({name:e,description:t,suggestionActions:n,inputSchema:r,capabilities:i})=>({name:e,description:t,suggestionActions:n,inputSchema:r,capabilities:i}))}async executeTool(e,t,n){let r=typeof n[W]==`string`?n[W]:void 0,i=typeof n[G]==`string`?n[G]:void 0;return this.telemetry.runInSpan(`browse_tool.custom_tool.execute`,{attributes:{"browse_tool.tool.name":t,"browse_tool.custom_tools.directory":A.resolve(e),"browse_tool.page.id":r,"browse_tool.browser.id":i}},async r=>{let i=(await this.loadTools(e)).find(e=>e.name===t);if(!i)throw r?.setStatus({code:P.ERROR,message:`Custom tool "${t}" not found`}),Error(`Custom tool "${t}" not found`);let{pageId:a,pageEntry:o,page:s,browser:c}=await this.resolveExecutionContext(t,n),l=this.createToolLogger(t,a,o);r?.setAttributes({"browse_tool.browser.id":o.browserId,"browse_tool.execution.mode":o.mode,"browse_tool.page.id":a}),bt(`Executing custom tool`,{toolName:t,directory:A.resolve(e),pageId:a,browserId:o.browserId,mode:o.mode,input:yt(n),scriptPath:i.scriptPath});let u;try{u=await i.execute?.({page:s,browser:c,input:n,logger:l})}catch(e){let n=e instanceof Error?e.message:String(e);throw bt(`Custom tool execution failed`,{toolName:t,pageId:a,browserId:o.browserId,mode:o.mode,error:n,stack:e instanceof Error?e.stack:void 0}),Error(`Custom tool "${t}" failed on page "${a}": ${n}`,{cause:e})}bt(`Custom tool execution completed`,{toolName:t,pageId:a,browserId:o.browserId,mode:o.mode,result:yt(u)});let d=Nt(t,u,i.suggestionActions),f=d.isError?d.content[0]?.text:void 0;return f&&r?.setStatus({code:P.ERROR,message:f}),d})}async loadTools(e){let t=A.resolve(e),n=A.join(t,`tools.yaml`),r=Dt(await M(n,`utf8`).catch(e=>{throw Error(`Failed to read custom tool manifest at "${n}": ${e instanceof Error?e.message:String(e)}`)})),i=_t.parse(r);return Promise.all(i.tools.map(async e=>{Ot(e);let n=A.resolve(t,e.script);await oe(n).catch(t=>{throw Error(`Custom tool "${e.name}" script not found at "${n}": ${t instanceof Error?t.message:String(t)}`)});let r=At(n,await kt(n));return{name:e.name,description:e.description,suggestionActions:e.suggestionActions,inputSchema:e.inputSchema,capabilities:e.capabilities,scriptPath:n,...r}}))}};function Ft(e){let t=new I;return t.get(`/browsers`,async t=>{try{let n=e.get(r.BrowserService),i=e.get(r.PageRegistry),a=n.listBrowsers(),o=i.list(),s=a.map(e=>{let t=o.filter(t=>t.browserId===e.id);return{id:e.id,profileName:e.profileName,currentPageId:e.currentPageId,createdAt:e.createdAt.toISOString(),pages:t.map(e=>({id:e.id,url:e.url,title:e.title,createdAt:e.createdAt.toISOString()}))}}),c={totalBrowsers:a.length,totalPages:o.length};return t.json({browsers:s,stats:c})}catch(e){return console.error(`Failed to list browsers:`,e),t.json({error:`Failed to list browsers`,message:e instanceof Error?e.message:String(e)},500)}}),t.delete(`/browsers`,async t=>{try{return await e.get(r.BrowserService).closeAll(),t.json({success:!0,message:`All browsers closed`})}catch(e){return console.error(`Failed to close all browsers:`,e),t.json({error:`Failed to close all browsers`,message:e instanceof Error?e.message:String(e)},500)}}),t.delete(`/browsers/:id`,async t=>{try{let n=t.req.param(`id`),i=e.get(r.BrowserService);return i.getBrowser(n)?(await i.closeBrowser(n),t.json({success:!0,message:`Browser "${n}" closed`})):t.json({error:`Browser "${n}" not found`},404)}catch(e){return console.error(`Failed to close browser:`,e),t.json({error:`Failed to close browser`,message:e instanceof Error?e.message:String(e)},500)}}),t.delete(`/pages/:id`,async t=>{try{let n=t.req.param(`id`),i=e.get(r.PageRegistry).get(n);return i?i.page?(await i.page.close(),t.json({success:!0,message:`Page "${n}" closed`})):t.json({error:`Page "${n}" is in extension mode and cannot be closed via API`},400):t.json({error:`Page "${n}" not found`},404)}catch(e){return console.error(`Failed to close page:`,e),t.json({error:`Failed to close page`,message:e instanceof Error?e.message:String(e)},500)}}),t}function It(e){return e.toLocaleString(`en-US`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1})}function Lt(e){let t=Math.floor(e/1e3),n=Math.floor(t/60),r=Math.floor(n/60);return r>0?`${r}h ${n%60}m`:n>0?`${n}m ${t%60}s`:`${t}s`}function Rt(e){return Lt(new Date().getTime()-e.getTime())}const zt=`table-container`,Bt=`stat-card`,Vt=`browser-row`,Ht=`page-row`,Y=`status-active`,Ut=`url-cell`,X=`timestamp-cell`,Wt=`actions-cell`,Z=`kill-btn`,Gt=`empty-state`;function Kt({browsers:e}){return e.length===0?L(`div`,{class:zt,children:R(`div`,{class:Gt,children:[L(`h3`,{children:`No Active Browsers`}),L(`p`,{children:`Launch a browser using MCP tools to see it here.`})]})}):L(`div`,{class:zt,children:R(`table`,{class:`browser-table`,children:[L(`thead`,{children:R(`tr`,{children:[L(`th`,{children:`ID`}),L(`th`,{children:`Status`}),L(`th`,{children:`URL / Title`}),L(`th`,{children:`Created`}),L(`th`,{children:`Age`}),L(`th`,{children:`Actions`})]})}),L(`tbody`,{id:`browser-table-body`,children:e.map(e=>R(be,{children:[R(`tr`,{class:Vt,children:[L(`td`,{children:e.id}),L(`td`,{children:R(`span`,{class:Y,children:[e.pages.length,` page`,e.pages.length===1?``:`s`]})}),L(`td`,{children:e.profileName||`Default Profile`}),L(`td`,{class:X,children:It(e.createdAt)}),L(`td`,{class:X,children:Rt(e.createdAt)}),L(`td`,{class:Wt,children:L(`button`,{type:`button`,class:Z,onclick:`dashboard.killBrowser('${e.id}')`,children:`Kill`})})]}),e.pages.map(t=>R(`tr`,{class:Ht,children:[R(`td`,{children:[t.id,e.currentPageId===t.id&&` (active)`]}),L(`td`,{children:L(`span`,{class:Y,children:`Open`})}),L(`td`,{class:Ut,title:t.url,children:t.title||t.url||`about:blank`}),L(`td`,{class:X,children:It(t.createdAt)}),L(`td`,{class:X,children:Rt(t.createdAt)}),L(`td`,{class:Wt,children:L(`button`,{type:`button`,class:Z,onclick:`dashboard.killPage('${t.id}')`,children:`Close`})})]},t.id))]},e.id))})]})})}function qt({stats:e}){return R(`div`,{class:`stats-container`,children:[R(`div`,{class:Bt,children:[L(`h3`,{children:`Active Browsers`}),L(`div`,{class:`value`,children:e.totalBrowsers})]}),R(`div`,{class:Bt,children:[L(`h3`,{children:`Active Pages`}),L(`div`,{class:`value`,children:e.totalPages})]})]})}function Jt({browsers:e,stats:t}){return R(`div`,{class:`dashboard-container`,children:[R(`div`,{class:`dashboard-header`,children:[R(`div`,{children:[L(`h1`,{children:`Playwright MCP Dashboard`}),L(`p`,{children:`Browser automation management (auto-refresh every 3 seconds)`})]}),R(`div`,{children:[L(`button`,{type:`button`,id:`refresh-btn`,class:`btn refresh-btn`,onclick:`dashboard.manualRefresh()`,children:`Refresh Now`}),L(`button`,{type:`button`,id:`kill-all-btn`,class:`btn btn-danger`,onclick:`dashboard.killAllBrowsers()`,children:`Kill All Browsers`})]})]}),L(qt,{stats:t}),L(Kt,{browsers:e}),L(`script`,{children:ye(`
6
6
  class DashboardManager {
7
7
  constructor() {
8
8
  this.autoRefreshInterval = null;
@@ -40,7 +40,7 @@ class DashboardManager {
40
40
  }
41
41
 
42
42
  updateStats(stats) {
43
- const cards = document.querySelectorAll('.${It} .value');
43
+ const cards = document.querySelectorAll('.${Bt} .value');
44
44
  if (cards.length >= 2) {
45
45
  cards[0].textContent = stats.totalBrowsers;
46
46
  cards[1].textContent = stats.totalPages;
@@ -52,10 +52,10 @@ class DashboardManager {
52
52
  if (!tbody) return;
53
53
 
54
54
  if (browsers.length === 0) {
55
- const container = tbody.closest('.${Ft}');
55
+ const container = tbody.closest('.${zt}');
56
56
  if (container) {
57
57
  container.innerHTML = \`
58
- <div class="${Ut}">
58
+ <div class="${Gt}">
59
59
  <h3>No Active Browsers</h3>
60
60
  <p>Launch a browser using MCP tools to see it here.</p>
61
61
  </div>
@@ -69,29 +69,29 @@ class DashboardManager {
69
69
  browsers.forEach(browser => {
70
70
  // Browser row
71
71
  const browserRow = document.createElement('tr');
72
- browserRow.className = '${Lt}';
72
+ browserRow.className = '${Vt}';
73
73
  browserRow.innerHTML = \`
74
74
  <td>\${browser.id}</td>
75
- <td><span class="${zt}">\${browser.pages.length} page\${browser.pages.length !== 1 ? 's' : ''}</span></td>
75
+ <td><span class="${Y}">\${browser.pages.length} page\${browser.pages.length !== 1 ? 's' : ''}</span></td>
76
76
  <td>\${browser.profileName || 'Default Profile'}</td>
77
- <td class="${Z}">\${this.formatTimestamp(browser.createdAt)}</td>
78
- <td class="${Z}">\${this.formatAge(browser.createdAt)}</td>
79
- <td class="${Vt}"><button class="${Ht}" onclick="dashboard.killBrowser('\${browser.id}')">Kill</button></td>
77
+ <td class="${X}">\${this.formatTimestamp(browser.createdAt)}</td>
78
+ <td class="${X}">\${this.formatAge(browser.createdAt)}</td>
79
+ <td class="${Wt}"><button class="${Z}" onclick="dashboard.killBrowser('\${browser.id}')">Kill</button></td>
80
80
  \`;
81
81
  tbody.appendChild(browserRow);
82
82
 
83
83
  // Page rows
84
84
  browser.pages.forEach(page => {
85
85
  const pageRow = document.createElement('tr');
86
- pageRow.className = '${Rt}';
86
+ pageRow.className = '${Ht}';
87
87
  const isActive = browser.currentPageId === page.id;
88
88
  pageRow.innerHTML = \`
89
89
  <td>\${page.id}\${isActive ? ' (active)' : ''}</td>
90
- <td><span class="${zt}">Open</span></td>
91
- <td class="${Bt}" title="\${this.escapeHtml(page.url)}">\${this.escapeHtml(page.title || page.url || 'about:blank')}</td>
92
- <td class="${Z}">\${this.formatTimestamp(page.createdAt)}</td>
93
- <td class="${Z}">\${this.formatAge(page.createdAt)}</td>
94
- <td class="${Vt}"><button class="${Ht}" onclick="dashboard.killPage('\${page.id}')">Close</button></td>
90
+ <td><span class="${Y}">Open</span></td>
91
+ <td class="${Ut}" title="\${this.escapeHtml(page.url)}">\${this.escapeHtml(page.title || page.url || 'about:blank')}</td>
92
+ <td class="${X}">\${this.formatTimestamp(page.createdAt)}</td>
93
+ <td class="${X}">\${this.formatAge(page.createdAt)}</td>
94
+ <td class="${Wt}"><button class="${Z}" onclick="dashboard.killPage('\${page.id}')">Close</button></td>
95
95
  \`;
96
96
  tbody.appendChild(pageRow);
97
97
  });
@@ -187,7 +187,7 @@ const dashboard = new DashboardManager();
187
187
  window.addEventListener('beforeunload', () => {
188
188
  dashboard.stopAutoRefresh();
189
189
  });
190
- `)})]})}function qt({title:e,children:t}){return z(`html`,{lang:`en`,children:[z(`head`,{children:[R(`meta`,{charset:`UTF-8`}),R(`meta`,{name:`viewport`,content:`width=device-width, initial-scale=1.0`}),R(`title`,{children:e}),R(`style`,{children:ge(`
190
+ `)})]})}function Yt({title:e,children:t}){return R(`html`,{lang:`en`,children:[R(`head`,{children:[L(`meta`,{charset:`UTF-8`}),L(`meta`,{name:`viewport`,content:`width=device-width, initial-scale=1.0`}),L(`title`,{children:e}),L(`style`,{children:ye(`
191
191
  * {
192
192
  margin: 0;
193
193
  padding: 0;
@@ -428,7 +428,7 @@ window.addEventListener('beforeunload', () => {
428
428
  margin-bottom: 0.5rem;
429
429
  color: #999;
430
430
  }
431
- `)})]}),R(`body`,{children:t})]})}function Jt(e){let t=new L;return t.get(`/`,async t=>{try{let n=e.get(r.BrowserService),i=e.get(r.PageRegistry),a=n.listBrowsers(),o=i.list(),s=a.map(e=>{let t=o.filter(t=>t.browserId===e.id);return{id:e.id,profileName:e.profileName,currentPageId:e.currentPageId,createdAt:e.createdAt,pages:t.map(e=>({id:e.id,url:e.url,title:e.title,createdAt:e.createdAt}))}}),c={totalBrowsers:a.length,totalPages:o.length};return t.html(R(qt,{title:`Playwright MCP Dashboard`,children:R(Kt,{browsers:s,stats:c})}))}catch(e){return console.error(`Failed to render dashboard:`,e),t.html(R(qt,{title:`Playwright MCP Dashboard - Error`,children:z(`div`,{style:`padding: 2rem; text-align: center;`,children:[R(`h1`,{children:`Failed to load dashboard`}),R(`p`,{children:e instanceof Error?e.message:String(e)}),R(`a`,{href:`/`,style:`color: #2196f3; text-decoration: underline;`,children:`Retry`})]})}),500)}}),t}function Yt(e){if(!e.isError)return;let t=e.content[0];return t?.type===`text`&&typeof t.text==`string`?t.text:`Unknown tool execution error`}function Xt(e){try{return e.get(r.TelemetryService)}catch{return new f}}function Zt(e){let t=new L,n=e.get(r.PageRegistry),i=e.get(r.BrowserService),a=e.get(r.ExtensionTaskQueue),o=Xt(e),s=new At(n,a,o,i);t.use(`*`,me({origin:e=>e??null,credentials:!0,allowMethods:[`GET`,`POST`,`DELETE`,`OPTIONS`],allowHeaders:[`Content-Type`,`Accept`,`Authorization`]})),t.get(`/health`,e=>{try{let t=i.listBrowsers(),n={status:`healthy`,service:`browse-tool-http`,serviceName:`browse-tool-http`,timestamp:new Date().toISOString(),browsers:{count:t.length,instances:t.map(e=>({id:e.id,pageCount:e.pageIds.size,createdAt:e.createdAt.toISOString()}))}};return e.json(n)}catch(t){return e.json({status:`unhealthy`,service:`browse-tool-http`,serviceName:`browse-tool-http`,error:t instanceof Error?t.message:String(t)},503)}}),t.post(`/execute`,async t=>{try{let i=await t.req.json();if(!i.tool)return t.json({success:!1,error:`Missing "tool" field in request body`},400);let a=typeof i.arguments?.pageId==`string`?i.arguments.pageId:void 0;return await o.runInSpan(`browse_tool.http.execute`,{attributes:{"http.method":`POST`,"http.route":`/execute`,"browse_tool.tool.name":i.tool,"browse_tool.page.id":a}},async a=>{let s=e.getAll(r.Tool).find(e=>e.getDefinition().name===i.tool);if(process.env.BROWSE_TOOL_DEBUG_EXTENSION_RECORDING===`1`&&i.tool===`browser_launch`){let e=typeof i.arguments?.videoDir==`string`?i.arguments.videoDir:typeof i.arguments?.video_dir==`string`?i.arguments.video_dir:``;console.log(`[ExtensionRecordingDebug] http execute browser_launch mode=${String(i.arguments?.mode??``)} videoDir=${e}`)}if(!s)return a?.setStatus({code:F.ERROR,message:`Tool "${i.tool}" not found`}),o.log(`warn`,`browse-tool HTTP execute rejected unknown tool`,{attributes:{"http.route":`/execute`,"browse_tool.tool.name":i.tool}}),t.json({success:!1,error:`Tool "${i.tool}" not found`},404);let c=await w(t.req.header(T),()=>s.execute(i.arguments||{})),l=i.arguments||{},u=typeof l.pageId==`string`?l.pageId:void 0;if(u){let t=e.get(r.BrowserService),i=n.get(u);i&&t.recordBrowserActivity(i.browserId,u)}let d=Yt(c);return d?(a?.setStatus({code:F.ERROR,message:d}),o.log(`error`,`browse-tool HTTP execute failed`,{attributes:{"http.route":`/execute`,"browse_tool.tool.name":i.tool,"browse_tool.page.id":u}})):o.log(`info`,`browse-tool HTTP execute succeeded`,{attributes:{"http.route":`/execute`,"browse_tool.tool.name":i.tool,"browse_tool.page.id":u}}),t.json({success:!c.isError,result:c,error:d})})}catch(e){return o.log(`error`,`browse-tool HTTP execute request crashed`,{attributes:{"http.route":`/execute`},exception:e}),t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/browsers`,t=>{try{let n=e.get(r.BrowserService).listBrowsers();return t.json({browsers:n.map(e=>({id:e.id,profileName:e.profileName,pageIds:Array.from(e.pageIds),currentPageId:e.currentPageId,createdAt:e.createdAt.toISOString()}))})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/tools`,t=>{try{let n=e.getAll(r.Tool);return t.json({tools:n.map(e=>e.getDefinition())})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/custom-tools`,async e=>{let t=e.req.query(`dir`);if(!t)return e.json({tools:[],error:`Missing "dir" query parameter`},400);try{let n=await s.listTools(t);return e.json({tools:n})}catch(t){return e.json({tools:[],error:t instanceof Error?t.message:String(t)},400)}}),t.post(`/custom-tools`,async t=>{let i=t.req.query(`dir`);if(!i)return t.json({success:!1,error:`Missing "dir" query parameter`},400);try{let a=await t.req.json();if(!a.tool)return t.json({success:!1,error:`Missing "tool" field in request body`},400);let c=typeof a.arguments?.pageId==`string`?a.arguments.pageId:void 0;return await o.runInSpan(`browse_tool.http.custom_tool.execute`,{attributes:{"http.method":`POST`,"http.route":`/custom-tools`,"browse_tool.tool.name":a.tool,"browse_tool.page.id":c,"browse_tool.custom_tools.directory":i}},async l=>{let u=await w(t.req.header(T),()=>s.executeTool(i,a.tool,a.arguments||{}));if(c){let t=e.get(r.BrowserService),i=n.get(c);i&&t.recordBrowserActivity(i.browserId,c)}let d=Yt(u);return d?(l?.setStatus({code:F.ERROR,message:d}),o.log(`error`,`browse-tool custom tool execution failed`,{attributes:{"http.route":`/custom-tools`,"browse_tool.tool.name":a.tool,"browse_tool.page.id":c}})):o.log(`info`,`browse-tool custom tool execution succeeded`,{attributes:{"http.route":`/custom-tools`,"browse_tool.tool.name":a.tool,"browse_tool.page.id":c}}),t.json({success:!u.isError,result:u,error:d})})}catch(e){return o.log(`error`,`browse-tool custom tool request crashed`,{attributes:{"http.route":`/custom-tools`,"browse_tool.custom_tools.directory":i},exception:e}),t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}});let c=xe(e);t.route(`/extension`,c);let l=jt(e);t.route(`/api`,l);let u=Jt(e);return t.route(`/`,u),t}function Qt(e,t,n){e.get(`/ws/extension/:browserId`,n(e=>{let n=e.req.param(`browserId`),i=null;return{onOpen(e,a){if(!n){a.close(1008,`Missing browserId`);return}try{i=t.get(r.WebSocketHub).addConnection(a,n)}catch{a.close(1011,`Internal server error`)}},onMessage(e){if(i)try{let n=t.get(r.WebSocketHub),a=typeof e.data==`string`?e.data:e.data.toString();n.handleMessage(i,a)}catch{}},onClose(){if(i)try{t.get(r.WebSocketHub).removeConnection(i)}catch{}},onError(){if(i)try{t.get(r.WebSocketHub).removeConnection(i)}catch{}}}})),e.get(`/ws/stats`,e=>{try{let n=t.get(r.WebSocketHub).getStats();return e.json({totalConnections:n.totalConnections,browserCount:n.browserCount,connections:n.connections.map(e=>({id:e.id,browserId:e.browserId,connectedAt:e.connectedAt.toISOString()}))})}catch(t){return e.json({error:t instanceof Error?t.message:String(t)},500)}})}const $t=[`pnpm-workspace.yaml`,`nx.json`,`.git`];function en(e=process.cwd()){let t=j.resolve(e);for(;;){for(let e of $t)if(M(j.join(t,e)))return t;let e=j.dirname(t);if(e===t)return process.cwd();t=e}}const tn=new I(`http-serve`).description(`Start HTTP server for browser automation`).option(`-p, --port <port>`,`Port to listen on`,String(u)).option(`--host <host>`,`Host to bind`,y()).option(`--headless`,`Run browsers in headless mode by default`,!0).option(`--idle-timeout <minutes>`,`Idle timeout in minutes before closing browsers`,`30`).option(`--registry-dir <path>`,`Custom registry path or directory for service discovery`).option(`--registry-path <path>`,`Custom registry path or directory for service discovery`).option(`--pids-dir <path>`,`Custom PIDs directory for process tracking (deprecated)`).option(`--profiles-dir <path>`,`Custom profiles directory for browser profiles`).option(`--snippets-dir <path>`,`Directory used by browser_run_code to save and load reusable snippets`).option(`--proxy-config-dir <path>`,`Custom proxy config directory`).action(async function(e){let t=V(`httpServe`),n={port:H(this,`port`,e.port,t.port===void 0?void 0:String(t.port),process.env.PLAYWRIGHT_PORT),headless:H(this,`headless`,e.headless,t.headless),idleTimeout:H(this,`idleTimeout`,e.idleTimeout,t.idleTimeout===void 0?void 0:String(t.idleTimeout),process.env[x]),host:H(this,`host`,e.host,t.host,process.env.PLAYWRIGHT_HOST),registryDir:H(this,`registryDir`,e.registryDir,t.registryDir,process.env.PLAYWRIGHT_REGISTRY_DIR),registryPath:H(this,`registryPath`,e.registryPath,t.registryPath,process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH),pidsDir:H(this,`pidsDir`,e.pidsDir,t.pidsDir,process.env.PLAYWRIGHT_PIDS_DIR),profilesDir:H(this,`profilesDir`,e.profilesDir,t.profilesDir,process.env.PLAYWRIGHT_PROFILES_DIR),snippetsDir:H(this,`snippetsDir`,e.snippetsDir,t.snippetsDir,process.env.BROWSE_TOOL_SNIPPETS_DIR),proxyConfigDir:H(this,`proxyConfigDir`,e.proxyConfigDir,t.proxyConfigDir,process.env[b])},i;try{let e=Number.parseInt(n.port,10),t=`browse-tool-http`,a=process.env.NODE_ENV||`development`,o=en(process.cwd()),c=n.registryPath||n.registryDir||process.env.PLAYWRIGHT_REGISTRY_PATH;c&&(process.env.PLAYWRIGHT_REGISTRY_DIR=c,process.env.PORT_REGISTRY_PATH=c,process.env.PROCESS_REGISTRY_PATH=te(c,`processes.json`)),process.env.PLAYWRIGHT_HOST=n.host,process.env.PLAYWRIGHT_PORT=n.port,process.env[x]=n.idleTimeout,n.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=n.pidsDir),n.profilesDir&&(process.env.PLAYWRIGHT_PROFILES_DIR=n.profilesDir),n.snippetsDir&&(process.env.BROWSE_TOOL_SNIPPETS_DIR=j.resolve(n.snippetsDir)),n.proxyConfigDir&&(process.env[b]=j.resolve(n.proxyConfigDir)),console.log(`Starting HTTP server for browser automation...`),console.log(` Port: ${e}`),console.log(` Host: ${n.host}`),console.log(` Headless: ${n.headless}`),console.log(` Idle Timeout: ${n.idleTimeout} minutes`),process.env.BROWSE_TOOL_SNIPPETS_DIR&&console.log(` Snippets Dir: ${process.env.BROWSE_TOOL_SNIPPETS_DIR}`),process.env.BROWSE_TOOL_PROXY_CONFIG_DIR&&console.log(` Proxy Config Dir: ${process.env[b]}`);let l=s(),u=l.get(r.ProcessRegistryService),f=Zt(l),{injectWebSocket:p,upgradeWebSocket:m}=he({app:f});Qt(f,l,m);let h=l.get(r.WebSocketHub),g=l.get(r.ExtensionTaskQueue),_=l.get(r.BrowserService),v=l.get(r.ExtensionSessionRegistry),y=l.get(r.PageRegistry),ee=l.get(r.IdleCleanupService);ee.start(),h.setEventHandlers({onSessionRegister:(e,t)=>{if(t.type===`session:register`){let n=t.payload.browserId||e.browserId,r=_.getBrowser(n),i,a;if(r&&(r.mode===`extension`||r.mode===`vm`)){i=r.id;let t=y.findByBrowser(i);a=r.currentPageId||t[0]?.id||``,h.reassignConnection(e.id,i),a||(a=y.registerExtensionPage(i),r.pageIds.add(a),r.currentPageId=a),console.log(`[WebSocket] Extension connected to existing browser: ${i}, pageId: ${a}`)}else{i=n,_.registerExtensionBrowserWithId(i),a=y.registerExtensionPage(i);let e=_.getBrowser(i);e&&(e.pageIds.add(a),e.currentPageId=a),console.log(`[WebSocket] Extension connected with new browser: ${i}, pageId: ${a}`)}let o=y.get(a);o&&(o.extensionTabId=t.payload.tabId,o.url=t.payload.url??o.url);let s=v.register({browserId:i,tabId:t.payload.tabId,url:t.payload.url,metadata:{transport:`websocket`}});h.sendSessionAck(e,t.id??``,s.id,s.controlMode),h.broadcastPageCreated(i,a,y.get(a)?.url)}},onTaskResult:(e,t)=>{if(t.type===`task:result`){let e=g.submitResult({taskId:t.payload.taskId,success:t.payload.success,result:t.payload.result,error:t.payload.error});e?.browserId&&_.recordBrowserActivity(e.browserId,e.pageId)}},onTabMapped:(e,t)=>{if(t.type===`tab:mapped`){let e=y.get(t.payload.pageId);if(!e)return;e.extensionTabId=t.payload.tabId,_.recordBrowserActivity(e.browserId,e.id)}},onDisconnect:e=>{e.sessionId&&v.removeSession(e.sessionId),console.log(`[WebSocket] Extension disconnected: ${e.browserId}`)}});let S=new O(process.env.PORT_REGISTRY_PATH);console.log(c?` Registry path: ${c}`:` Registry path: default (~/.port-registry/ports.json)`);let C=await S.reservePort({repositoryPath:o,serviceName:t,serviceType:`tool`,environment:a,preferredPort:e,pid:process.pid,host:n.host,force:!0,portRange:D,metadata:{headless:n.headless,idleTimeout:n.idleTimeout}});if(!C.success||C.port===void 0)throw Error(C.error||`Failed to reserve port in range ${D.min}-${D.max}`);let w=C.port;C.record.metadata={...C.record.metadata,healthCheckUrl:`${d(n.host,w)}/health`};let T=pe({fetch:f.fetch,port:w,hostname:n.host});p(T),i=await k({repositoryPath:o,serviceName:t,serviceType:`service`,environment:a,pid:process.pid,port:w,host:n.host,metadata:{healthCheckUrl:`${d(n.host,w)}/health`,headless:n.headless,idleTimeout:n.idleTimeout,transport:`http`}},u);let E=d(n.host,w);console.log(`HTTP server listening on ${E}`),console.log(` Health check: ${E}/health`),console.log(` Execute tool: POST ${E}/execute`),console.log(`
432
- Press Ctrl+C to stop the server`);let A=async e=>{console.log(`\n\n${e} received. Shutting down gracefully...`);try{ee.stop(),T.close(),console.log(`Server closed`);try{await l.get(r.BrowserService).closeAll(),console.log(`All browsers closed`)}catch{}try{i&&await i.release({kill:!1}),console.log(`Service deregistered`)}catch(e){console.error(`Process registry cleanup error:`,e)}console.log(`Goodbye!`),process.exit(0)}catch(e){console.error(`Error during shutdown:`,e),process.exit(1)}};process.on(`SIGINT`,()=>A(`SIGINT`)),process.on(`SIGTERM`,()=>A(`SIGTERM`)),await new Promise(()=>{})}catch(e){if(i)try{await i.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}console.error(`Error starting HTTP server:`,e),process.exit(1)}}),nn={browser_click:[`input`],browser_fill:[`input`],browser_type:[`input`],browser_select:[`input`],browser_hover:[`input`],browser_drag:[`input`],browser_press_key:[`input`],browser_upload_file:[`input`],browser_navigate:[`navigation`],browser_go_back:[`navigation`],browser_go_forward:[`navigation`],browser_reload:[`navigation`],browser_wait_for:[`navigation`],browser_snapshot:[`snapshot`],browser_screenshot:[`snapshot`],browser_pdf:[`snapshot`],browser_list_pages:[`page`],browser_new_page:[`page`],browser_select_page:[`page`],browser_close_page:[`page`],browser_resize_page:[`page`,`emulation`],browser_handle_dialog:[`dialog`],browser_list_network_requests:[`network`],browser_get_network_request:[`network`],browser_list_console_messages:[`console`],browser_evaluate_script:[`script`],browser_emulate:[`emulation`],browser_expect:[`testing`],browser_start_trace:[`tracing`],browser_stop_trace:[`tracing`],browser_list_profiles:[`profile`],browser_delete_profile:[`profile`],run_spec:[`spec`,`testing`],discover_specs:[`spec`,`testing`],browser_launch:[`browser`],browser_close:[`browser`],browser_run_code:[`code`,`script`]};function rn(e){let t=new Set;for(let[n,r]of Object.entries(nn))r.some(t=>e.includes(t))&&t.add(n);return t}const an=3e3,on=[`browser_launch`],sn=[`browser_new_page`],cn=[`browser_close`],ln=[`browser_close_page`];function un(e){if(!e?.content?.[0]?.text)return null;try{let t=JSON.parse(e.content[0].text);return{browserId:t.browserId,pageId:t.pageId,mode:t.mode,url:t.url,closedBrowserId:t.closedBrowserId,closedPageId:t.closedPageId,browserKeptOpen:t.browserKeptOpen}}catch{return null}}async function dn(e){await new Promise(t=>setTimeout(t,e))}async function fn(e){let t=new AbortController,n=setTimeout(()=>t.abort(),an);try{let n=await fetch(`${e}/tools`,{signal:t.signal});if(!n.ok)throw Error(`HTTP error ${n.status}: ${n.statusText}`);let r=await n.json();if(r.error)throw Error(r.error);if(!Array.isArray(r.tools))throw Error(`Invalid /tools response: missing tools array`);if(r.tools.length===0)throw Error(`HTTP server returned an empty tool list`);return r.tools}finally{clearTimeout(n)}}function pn(e,t){let n=new URL(`/custom-tools`,e);return n.searchParams.set(`dir`,t),n.toString()}async function mn(e,t){let n=new AbortController,r=setTimeout(()=>n.abort(),an);try{let r=await fetch(pn(e,t),{signal:n.signal});if(!r.ok)throw Error(`HTTP error ${r.status}: ${r.statusText}`);let i=await r.json();if(i.error)throw Error(i.error);if(!Array.isArray(i.tools))throw Error(`Invalid /custom-tools response: missing tools array`);return i.tools}finally{clearTimeout(r)}}async function hn(e){let t;for(let n=1;n<=4;n+=1)try{return await fn(e)}catch(e){t=e instanceof Error?e:Error(String(e)),n<4&&await dn(150*n)}throw Error(`Failed to fetch tools after 4 attempts: ${t?.message??`Unknown error`}`,{cause:t})}async function gn(e,t){let n;for(let r=1;r<=4;r+=1)try{return await mn(e,t)}catch(e){n=e instanceof Error?e:Error(String(e)),r<4&&await dn(150*r)}throw Error(`Failed to fetch custom tools after 4 attempts: ${n?.message??`Unknown error`}`,{cause:n})}function _n(e){if(!e?.tags?.length&&!e?.exclude?.length)return null;let t=e.tags?.length?rn(e.tags):null,n=new Set(e.exclude??[]);if(!t)return n.size>0?n:null;for(let e of n)t.delete(e);return t}function vn(e){let t=new Map,n=(e,n)=>{let r=t.get(e)??new Set;r.add(n),t.set(e,r)};return e.defaultMode&&n(`browser_launch`,`mode`),e.enforcedProfileName&&n(`browser_launch`,`profileName`),t}function yn(e,t){let n=t.get(e.name);if(!n?.size)return e;let r=e.inputSchema;if(!r.properties||typeof r.properties!=`object`)return e;let i=Object.fromEntries(Object.entries(r.properties).filter(([e])=>!n.has(e))),a={...r,properties:i};return Array.isArray(r.required)&&(a.required=r.required.filter(e=>typeof e!=`string`||!n.has(e))),{...e,inputSchema:a}}function bn(e){let{httpBaseUrl:t,sessionTracker:n,defaultMode:r,toolFilter:i,customToolsDir:a,enforcedProfileName:o,proxyConfigDir:s,ownerId:c}=e,u=i?.tags?.length?_n(i):null,d=new Set(i?.exclude??[]),f=u!==null||d.size>0,p=vn({defaultMode:r,enforcedProfileName:o}),m=new ce({name:`browse-tool`,version:`0.1.0`},{capabilities:{tools:{}}}),h={name:`browser_list_session`,description:`List all browsers and pages created in this MCP session. Use this to see what resources you have available.`,inputSchema:{type:`object`,properties:{},required:[],additionalProperties:!1}},g=[],_=[],v=new Set;function y(e){return f?e.filter(e=>d.has(e.name)?!1:u===null?!0:u.has(e.name)):e}function b(e){return e.filter(e=>!d.has(e.name))}function ee(e){return o?{...e,profileName:o}:e}function x(e){let t=l({proxy:e.proxy,profileName:typeof e.profileName==`string`?e.profileName:void 0,proxyConfigDir:s});if(!t){let t={...e};return delete t.proxy,t}return{...e,proxy:t}}function S(e){if(!o)return e;let t=e?.content?.[0]?.text;if(!t)return e;try{let n=JSON.parse(t),r=(Array.isArray(n.profiles)?n.profiles:[]).filter(e=>e?.name===o);return{...e,content:[{...e.content[0],text:JSON.stringify({...n,profileCount:r.length,profiles:r},null,2)},...e.content.slice(1)]}}catch{return e}}function C(e){return{name:e.name,description:e.description,inputSchema:e.inputSchema,annotations:e.capabilities}}function w(e){return d.has(e)?!1:u===null?!0:u.has(e)}return m.setRequestHandler(ue,async()=>{try{let e=await hn(t);g=e,v=new Set(e.map(e=>e.name));let r=a?await gn(t,a):[];_=r;let i=[...y(e).map(e=>yn(e,p)),...b(r).map(e=>C(e))];return n&&i.push(h),{tools:i}}catch(e){if(g.length>0||_.length>0){console.error(`Failed to refresh tools from HTTP server. Returning cached tools:`,e instanceof Error?e.message:String(e));let t=[...y(g).map(e=>yn(e,p)),...b(_).map(e=>C(e))];return n&&t.push(h),{tools:t}}let r=e instanceof Error?e.message:String(e);throw Error(`Failed to list tools from HTTP server at ${t}: ${r}`,{cause:e})}}),m.setRequestHandler(le,async e=>{let{name:i,arguments:o}=e.params,s=new Set(_.map(e=>e.name));if(i!==`browser_list_session`&&!w(i))return{content:[{type:`text`,text:`Tool "${i}" is not available (filtered by --tags or --exclude)`}],isError:!0};if(i===`browser_list_session`&&n){let e=n.getSessionState();return{content:[{type:`text`,text:JSON.stringify(e,null,2)}]}}let l=o||{};i===`browser_launch`&&r&&l.mode===void 0&&(l={...l,mode:r}),i===`browser_launch`&&(l=ee(l),l=x(l));try{if(a&&!s.has(i)&&!v.has(i))try{_=await gn(t,a),s=new Set(_.map(e=>e.name))}catch(e){console.error(`Failed to refresh custom tools, using cached list: ${e instanceof Error?e.message:String(e)}`)}let e=s.has(i),r=await fetch(e&&a?pn(t,a):`${t}/execute`,{method:`POST`,headers:c?{"Content-Type":`application/json`,[T]:c}:{"Content-Type":`application/json`},body:JSON.stringify({tool:i,arguments:l})});if(!r.ok){let e=await r.text();return{content:[{type:`text`,text:`HTTP error ${r.status}: ${e}`}],isError:!0}}let o=await r.json();if(!o.success)return{content:[{type:`text`,text:o.error||o.result?.content?.[0]?.text||`Unknown error from HTTP server`}],isError:!0};if(n&&o.result){if(on.includes(i)){let e=un(o.result);e?.browserId&&(n.trackBrowser(e.browserId,e.mode||`playwright`),e.pageId&&n.trackPage(e.pageId,e.browserId,e.url))}else if(sn.includes(i)){let e=un(o.result);e?.pageId&&e?.browserId&&n.trackPage(e.pageId,e.browserId,e.url)}else if(cn.includes(i)){let e=un(o.result);e?.closedBrowserId&&n.untrackBrowser(e.closedBrowserId)}else if(ln.includes(i)){let e=un(o.result);e?.closedPageId&&n.untrackPage(e.closedPageId),e?.browserId&&e.browserKeptOpen===!1&&n.untrackBrowser(e.browserId)}}return i===`browser_list_profiles`&&o.result?S(o.result):o.result||{content:[{type:`text`,text:`No result returned`}]}}catch(e){return{content:[{type:`text`,text:`Failed to execute tool via HTTP server: ${e instanceof Error?e.message:String(e)}`}],isError:!0}}}),m}const Q=`stdio`,xn=`http`,$=`streamable-http`,Sn=new Set([Q,xn,$]),Cn=`chromium`,wn=new Set([Cn,`firefox`,`webkit`]),Tn=new Set([`playwright`,`extension`,`vm`,`stealth`]),En=`SIGINT`,Dn=`SIGTERM`,On=`PLAYWRIGHT_PORT`,kn=[`pnpm-workspace.yaml`,`nx.json`,`.git`],An=`browse-tool-mcp-http`,jn=`service`,Mn=D.min;var Nn=class extends Error{code=`INVALID_TRANSPORT`;recovery=`Use --type ${[...Sn].join(`, `)}`;transportType;constructor(e,t){super(`Unknown transport type: ${e}`,t),this.name=`InvalidTransportError`,this.transportType=e}},Pn=class extends Error{code=`INVALID_BROWSER_TYPE`;recovery=`Use --browser ${[...wn].join(`, `)}`;browserType;constructor(e,t){super(`Unknown browser type: ${e}`,t),this.name=`InvalidBrowserTypeError`,this.browserType=e}},Fn=class extends Error{code=`INVALID_LAUNCH_MODE`;recovery=`Use --mode ${[...Tn].join(`, `)}`;launchMode;constructor(e,t){super(`Unknown launch mode: ${e}`,t),this.name=`InvalidLaunchModeError`,this.launchMode=e}};function In(e){let t=e.toLowerCase();if(!wn.has(t))throw new Pn(e);return t}function Ln(e){let t=e.toLowerCase();if(!Tn.has(t))throw new Fn(e);return t}function Rn(e){let t=typeof e==`number`?e:Number.parseInt(e,10);if(!Number.isInteger(t)||t<=0||t>65535)throw Error(`Invalid port: ${e}`);return t}function zn(e){return e!=null&&String(e).trim()!==``}function Bn(e){let t=e.toLowerCase();return t===xn?$:t}function Vn(e,t){let n=e[t];return(Array.isArray(n)?n[0]:n)?.trim()||void 0}function Hn(e=process.cwd()){let t=j.resolve(e);for(;;){for(let e of kn)if(M(j.join(t,e)))return t;let e=j.dirname(t);if(e===t)return process.cwd();t=e}}function Un(e){return e.split(`,`).map(e=>e.trim()).filter(e=>e.length>0)}function Wn(e){let t=e.tags?Un(e.tags):void 0,n=e.exclude?Un(e.exclude):void 0;if(!(!t?.length&&!n?.length))return{tags:t,exclude:n}}var Gn=class extends Error{code=`BROWSER_CLEANUP_FAILED`;recovery=`Browsers may still be running; check the HTTP server or kill processes manually`;failedBrowserIds;httpBaseUrl;constructor(e,t,n){super(`Failed to close ${e.length} browser(s): ${e.join(`, `)}`,n),this.name=`BrowserCleanupError`,this.failedBrowserIds=e,this.httpBaseUrl=t}},Kn=class extends Error{code=`SESSION_SHUTDOWN_FAILED`;recovery=`Check logs for cleanup and transport errors; processes may need manual cleanup`;signal;constructor(e,t){super(`Shutdown failed for signal ${e}`,t),this.name=`SessionShutdownError`,this.signal=e}},qn=class extends Error{code=`HTTP_SERVER_START_FAILED`;recovery=`Check if the HTTP server binary exists and the port is available`;constructor(e){super(`Failed to start HTTP server`,e),this.name=`HttpServerStartError`}},Jn=class extends Error{code=`MCP_SERVER_BOOTSTRAP_FAILED`;recovery=`Check if all dependencies are installed and try again`;constructor(e){super(`Failed to start MCP server`,e),this.name=`McpServerBootstrapError`}};async function Yn(e,t,n){let r=e.getBrowserIds(),i=[],a;for(let e of r)try{let r=await fetch(`${t}/execute`,{method:`POST`,headers:n?{"Content-Type":`application/json`,[T]:n}:{"Content-Type":`application/json`},body:JSON.stringify({tool:`browser_close`,arguments:{browserId:e}})});if(!r.ok){let e=await r.text().catch(()=>``);throw Error(`HTTP ${r.status} from ${t}/execute: ${e}`)}console.error(` Closed browser: ${e}`)}catch(t){let n=t instanceof Error?t:Error(String(t));console.error(` Failed to close browser: ${e}`,n),i.push(e),a??=n}if(i.length>0)throw new Gn(i,t,{cause:a});e.clear()}async function Xn(e,t,n,r,i){await e.start();let a=!1,o=async o=>{if(a)return;a=!0,console.error(`\nReceived ${o}, shutting down gracefully...`);let s;try{let e=t.getSessionState();e.totalBrowsers>0&&(console.error(` Cleaning up ${e.totalBrowsers} browser(s) from this session...`),await Yn(t,n,r))}catch(e){s=e instanceof Error?e:Error(String(e)),console.error(`Browser cleanup error:`,s)}finally{try{await e.stop()}catch(e){let t=e instanceof Error?e:Error(String(e));console.error(`Transport stop error:`,t),s??=t}try{await i?.()}catch(e){let t=e instanceof Error?e:Error(String(e));console.error(`Process registry cleanup error:`,t),s??=t}if(s){let e=new Kn(o,{cause:s});console.error(`Error during shutdown:`,e),process.exit(1)}process.exit(0)}};process.once(En,()=>o(En)),process.once(Dn,()=>o(Dn))}const Zn=new I(`mcp-serve`).description(`Start Playwright MCP server for browser automation`).option(`-t, --type <type>`,`Transport type: ${[Q,$].join(`, `)}`,Q).option(`-b, --browser <browser>`,`Default browser type: ${[...wn].join(`, `)}`,Cn).option(`--headless`,`Run browsers in headless mode by default`,!1).option(`--no-headless`,`Run browsers in headed mode (visible window)`).option(`--host <host>`,`Default host for HTTP services`,y()).option(`--port <port>`,`Port for streamable HTTP transport`,String(Mn)).option(`--http-port <port>`,`Port for the inner browse-tool HTTP service`).option(`-p, --profile <name>`,`Default profile name for browser sessions`).option(`-m, --mode <mode>`,`Default launch mode: ${[...Tn].join(`, `)}`).option(`--tags <tags>`,`Comma-separated tags to filter tools (e.g. input,navigation,snapshot)`).option(`--exclude <tools>`,`Comma-separated tool names to exclude (applied after --tags)`).option(`--custom-tools <path>`,`Path to a folder containing tools.yaml and custom tool scripts`).option(`--chrome-for-testing-path <path>`,`Path to a Chrome for Testing executable for the inner HTTP server`).option(`--snippets-dir <path>`,`Path to a folder where browser_run_code snippets are stored`).option(`--idle-timeout <minutes>`,`Idle timeout in minutes before auto-closing browsers`).option(`--registry-path <path>`,`Custom registry path or directory for service discovery`).option(`--registry-dir <path>`,`Custom registry directory for service discovery`).option(`--pids-dir <path>`,`Custom PIDs directory for process tracking`).option(`--profiles-dir <path>`,`Custom profiles directory for browser profiles`).option(`--proxy-config-dir <path>`,`Custom proxy config directory`).action(async function(e){let t=V(`mcpServe`),n={type:H(this,`type`,e.type,t.type),browser:H(this,`browser`,e.browser,t.browser),headless:H(this,`headless`,e.headless,t.headless),profile:H(this,`profile`,e.profile,t.profile),mode:H(this,`mode`,e.mode,t.mode),host:H(this,`host`,e.host,t.host,process.env.PLAYWRIGHT_HOST),port:H(this,`port`,e.port,t.port,process.env.PLAYWRIGHT_MCP_PORT),httpPort:H(this,`httpPort`,e.httpPort,t.httpPort,process.env[On]),tags:H(this,`tags`,e.tags,t.tags),exclude:H(this,`exclude`,e.exclude,t.exclude),customTools:H(this,`customTools`,e.customTools,t.customTools),chromeForTestingPath:H(this,`chromeForTestingPath`,e.chromeForTestingPath,t.chromeForTestingPath,process.env[C]),snippetsDir:H(this,`snippetsDir`,e.snippetsDir,t.snippetsDir),idleTimeout:H(this,`idleTimeout`,e.idleTimeout,t.idleTimeout===void 0?void 0:String(t.idleTimeout),process.env[x]),registryPath:H(this,`registryPath`,e.registryPath,t.registryPath,process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH),registryDir:H(this,`registryDir`,e.registryDir,t.registryDir,process.env.PLAYWRIGHT_REGISTRY_DIR),pidsDir:H(this,`pidsDir`,e.pidsDir,t.pidsDir,process.env.PLAYWRIGHT_PIDS_DIR),profilesDir:H(this,`profilesDir`,e.profilesDir,t.profilesDir,process.env.PLAYWRIGHT_PROFILES_DIR),proxyConfigDir:H(this,`proxyConfigDir`,e.proxyConfigDir,t.proxyConfigDir,process.env[b])},i=Bn(n.type),a=Hn(process.cwd()),o=process.env.NODE_ENV||`development`,s,c;try{if(!Sn.has(i))throw new Nn(i);let e=In(n.browser),t=n.mode?Ln(n.mode):void 0,l=Rn(n.port??Mn),u=zn(n.httpPort),f=u?Rn(n.httpPort):void 0,p=Wn(n),m=n.customTools?j.resolve(n.customTools):void 0,g=n.chromeForTestingPath?j.resolve(n.chromeForTestingPath):void 0,S=n.snippetsDir?j.resolve(n.snippetsDir):void 0;console.error(`Playwright MCP Server starting...`),console.error(` Transport: ${i}`),i===$&&console.error(` MCP endpoint: http://${n.host??y()}:${l}/mcp`),console.error(` Default browser: ${e}`),console.error(` Headless: ${n.headless}`),t&&console.error(` Default mode: ${t}`),n.profile&&console.error(` Profile: ${n.profile}`),p?.tags?.length&&console.error(` Tags: ${p.tags.join(`, `)}`),p?.exclude?.length&&console.error(` Exclude: ${p.exclude.join(`, `)}`),m&&console.error(` Custom tools: ${m}`),g&&console.error(` Chrome for Testing path: ${g}`),S&&console.error(` Snippets dir: ${S}`),n.idleTimeout&&console.error(` Idle timeout: ${n.idleTimeout} minutes`);let w=n.registryPath||n.registryDir;w&&(process.env.PLAYWRIGHT_REGISTRY_DIR=w,process.env.PORT_REGISTRY_PATH=w,process.env.PROCESS_REGISTRY_PATH=te(w,`processes.json`),console.error(` Registry path: ${w}`)),n.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=n.pidsDir,console.error(` PIDs dir: ${n.pidsDir}`)),n.profilesDir&&(process.env.PLAYWRIGHT_PROFILES_DIR=n.profilesDir,console.error(` Profiles dir: ${n.profilesDir}`)),n.proxyConfigDir&&(process.env[b]=j.resolve(n.proxyConfigDir),console.error(` Proxy config dir: ${process.env[b]}`)),S&&(process.env.BROWSE_TOOL_SNIPPETS_DIR=S),g&&(process.env[C]=g),n.idleTimeout&&(process.env[x]=n.idleTimeout),n.host&&(process.env.PLAYWRIGHT_HOST=n.host);let T=v(),E=T.get(r.ProcessRegistryService);c=await T.get(r.McpPortAllocationService).acquirePorts({repositoryPath:a,environment:o,host:n.host??y(),pid:process.pid,preferredMcpPort:i===$?l:void 0,preferredHttpPort:f,exactHttpPort:u,reserveMcpPort:i===$,mcpServiceName:An,mcpServiceType:`tool`,mcpMetadata:i===$?{healthCheckUrl:`${d(n.host??y(),l)}/health`,transport:$}:void 0});let D=c.httpPort;process.env[On]=D.toString();let O=await T.get(r.HttpServerManager).ensureRunning(D,{exactPort:!0});if(!O.running)throw new qn;let A=d(y(),O.port),ne=O.spawned?`spawned`:`reused`;if(console.error(` HTTP server: ${ne} on port ${O.port}`),i===Q){let r=new h,i=se(),l=new _(bn({httpBaseUrl:A,sessionTracker:r,defaultMode:t,toolFilter:p,customToolsDir:m,enforcedProfileName:n.profile,proxyConfigDir:process.env[b],ownerId:i}));s=await k({repositoryPath:a,serviceName:`browse-tool-mcp`,serviceType:jn,environment:o,pid:process.pid,metadata:{transport:Q,browser:e,mode:t}},E),await Xn(l,r,A,i,async()=>{await s?.release({kill:!1}),await c?.release()});return}let M=c.mcpPort;if(!M)throw Error(`Failed to reserve ${$} port ${l}`);s=await k({repositoryPath:a,serviceName:An,serviceType:jn,environment:o,pid:process.pid,port:M,host:n.host??y(),metadata:{healthCheckUrl:`${d(n.host??y(),M)}/health`,transport:$}},E);let re=new ee(({headers:e})=>{let r=Vn(e,`x-profile`)??n.profile,i=new h,a=se();return{server:bn({httpBaseUrl:A,sessionTracker:i,defaultMode:t,toolFilter:p,customToolsDir:m,enforcedProfileName:r,proxyConfigDir:process.env[b],ownerId:a}),onClose:async()=>{i.getSessionState().totalBrowsers>0&&await Yn(i,A,a)}}},{host:n.host??y(),port:M});try{await re.start()}catch(e){try{await s.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}throw await c.release(),c=void 0,e}let ie=!1,N=async e=>{if(!ie){ie=!0,console.error(`\nReceived ${e}, shutting down gracefully...`);try{await re.stop();try{await s?.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}await c?.release(),c=void 0,process.exit(0)}catch(e){console.error(`Transport stop error:`,e),process.exit(1)}}};process.once(En,()=>N(En)),process.once(Dn,()=>N(Dn))}catch(e){if(s)try{await s.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}if(c)try{await c.release()}catch(e){console.error(`Port allocation cleanup error:`,e)}(e instanceof Nn||e instanceof Pn||e instanceof Fn||e instanceof qn)&&(console.error(`Error [${e.code}]: ${e.message}`),console.error(`Recovery: ${e.recovery}`),process.exit(1));let t=new Jn({cause:e instanceof Error?e:Error(String(e))});console.error(`Error [${t.code}]: ${t.message}`,t.cause),console.error(`Recovery: ${t.recovery}`),process.exit(1)}}),Qn=new I(`status`).description(`Show status of HTTP server and diagnostics`).action(async()=>{try{let e=V(`httpServe`),t=process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH??e.registryPath??e.registryDir;t&&(process.env.PLAYWRIGHT_REGISTRY_DIR=t,process.env.PLAYWRIGHT_REGISTRY_PATH=t,process.env.PORT_REGISTRY_PATH=t),!process.env.PLAYWRIGHT_PIDS_DIR&&e.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=e.pidsDir),console.log(`browse-tool Status
433
- `),console.log(`-`.repeat(50));let n=await m().get(r.HttpServerManager).getStatus();if(n.running){let t=process.env.PLAYWRIGHT_HOST??e.host??y();console.log(`HTTP Server: Running`),console.log(` PID: ${n.pid}`),console.log(` Port: ${n.port}`),console.log(` Health: ${d(t,n.port)}/health`),n.browserCount!==void 0&&console.log(` Browsers: ${n.browserCount}`)}else console.log(`HTTP Server: Not Running`),n.error&&console.log(` Error: ${n.error}`);console.log(``),console.log(`-`.repeat(50)),process.exit(0)}catch(e){console.error(`Error getting status:`,e),process.exit(1)}}),$n=new I(`stop`).description(`Stop HTTP server and clean up registry/PID files`).action(async()=>{try{let e=V(`httpServe`),t=process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH??e.registryPath??e.registryDir;t&&(process.env.PLAYWRIGHT_REGISTRY_DIR=t,process.env.PLAYWRIGHT_REGISTRY_PATH=t,process.env.PORT_REGISTRY_PATH=t),!process.env.PLAYWRIGHT_PIDS_DIR&&e.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=e.pidsDir),console.log(`Stopping browse-tool services...`),await m().get(r.HttpServerManager).stop()?(console.log(`HTTP server stopped`),console.log(`Registry and PID files cleaned up`)):console.log(`No HTTP server was running`),console.log(`Done!`),process.exit(0)}catch(e){console.error(`Error stopping services:`,e),process.exit(1)}}),er=new Set([`--config`]);function tr(e){for(let t=0;t<e.length;t+=1){let n=e[t];if(er.has(n)){t+=1;continue}if(![...er].some(e=>n.startsWith(`${e}=`))&&!n.startsWith(`-`))return n}}function nr(e,t=!1){if(t)return!1;let n=tr(e);return n?n===`tools`?!0:n===`help`&&e.includes(`tools`):!1}function rr(e){return e.replace(/_/g,`-`)}function ir(e){return e.replace(/([a-z])([A-Z])/g,`$1-$2`).toLowerCase()}function ar(e){let t=new Set;e.type&&t.add(e.type);for(let n of e.oneOf??[])n.type&&t.add(n.type);for(let n of e.anyOf??[])n.type&&t.add(n.type);return t}function or(e,t){let n=ar(t);if(n.has(`boolean`)){if(e===`true`||e===`1`)return!0;if(e===`false`||e===`0`)return!1}if(t.type===`number`||t.type===`integer`){let t=Number(e);if(Number.isNaN(t))throw Error(`Invalid number: ${e}`);return t}if(t.type===`boolean`){if(e===`true`||e===`1`)return!0;if(e===`false`||e===`0`)return!1;throw Error(`Invalid boolean: ${e}`)}if(n.has(`array`)||n.has(`object`))try{return JSON.parse(e)}catch{if(t.type===`array`||t.type===`object`)throw Error(`Invalid JSON: ${e}`)}if(n.has(`number`)||n.has(`integer`)){let t=Number(e);if(!Number.isNaN(t)&&e.trim()!==``)return t}return e}function sr(e,t){let n=ir(e),r=ar(t);return t.type===`boolean`?`--${n}`:r.has(`boolean`)&&r.has(`object`)?`--${n} [value]`:`--${n} <value>`}function cr(e){let{definition:t,execute:n}=e,r=new I(rr(t.name));r.description(t.description);let i=t.inputSchema,a=i.properties||{},o=new Set(i.required||[]);for(let[e,t]of Object.entries(a)){let n=t,i=sr(e,n),a=n.description||``;n.enum&&n.enum.length>0&&(a+=` (choices: ${n.enum.join(`, `)})`),n.default!==void 0&&(a+=` (default: ${JSON.stringify(n.default)})`),o.has(e)&&(a+=` [required]`),n.type===`boolean`?n.default===!0?r.option(`--no-${ir(e)}`,`Disable ${a}`):r.option(i,a,n.default):n.enum&&n.enum.length>0?r.addOption(new fe(i,a).choices(n.enum)):r.option(i,a)}return r.action(async function(){let e=this.optsWithGlobals(),r=V(`tools`),i=H(this,`format`,e.format??`json`,r.format),s=H(this,`color`,e.color??!0,r.color),c=H(this,`port`,String(e.port??u),r.port===void 0?void 0:String(r.port),process.env.PLAYWRIGHT_PORT),l={format:i,color:s},d=Je(t.name);try{let r={};for(let[t,n]of Object.entries(a)){let i=n,a=ir(t).replace(/-([a-z])/g,(e,t)=>t.toUpperCase()),o=this.getOptionValueSourceWithGlobals(a),s=e[a];i.type===`boolean`&&i.default===!0&&(s=e[a]),(o===void 0||o===`default`||o===`implied`)&&d[t]!==void 0&&(s=d[t]),s===void 0&&i.default!==void 0&&(s=i.default),s!==void 0&&(typeof s==`string`&&i.type!==`string`?r[t]=or(s,i):r[t]=s)}for(let e of o)if(r[e]===void 0){let t=ir(e);console.error(G(`Missing required option: --${t}`,l.color)),process.exit(1)}let i=Number.parseInt(c,10),s=U({port:i}),u=n?await n(r,{command:this,formatterOptions:l,port:i}):await s.execute(t.name,r),f=Ze(u,l);f&&console.log(f),u.isError&&process.exit(1)}catch(e){console.error(G(e instanceof Error?e:String(e),l.color)),process.exit(1)}}),r}function lr(e){return e.map(cr)}const ur={name:`browser_list_session`,description:`List browsers and pages currently tracked by the browse-tool HTTP server.`,inputSchema:{type:`object`,properties:{},required:[],additionalProperties:!1}};function dr(){return new I(`tools`).description(`Execute browser automation tools`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u))}function fr(e){return{browserCount:e.length,browsers:e.map(e=>({browserId:e.id,profileName:e.profileName,currentPageId:e.currentPageId,pageIds:e.pageIds,createdAt:e.createdAt}))}}function pr(){return{definition:ur,execute:async(e,t)=>{let n=await U({port:t.port}).listBrowsers();return{content:[{type:`text`,text:JSON.stringify(fr(n),null,2)}]}}}}function mr(e){return[...e.map(e=>({definition:e})),pr()]}async function hr(e,t){let n=V(`tools`),r=Number(t?.port??process.env.PLAYWRIGHT_PORT??n.port??u);try{let t=lr(mr(await U({port:r}).listTools()));for(let n of t)e.addCommand(n)}catch(e){let t=e instanceof Error?e.message:String(e);process.stderr.write(`${G(`Failed to load tool commands: ${t}`,!0)}\n`),process.stderr.write(`Make sure the HTTP server is running: browse-tool http-serve
434
- `)}}function gr(e,t){let n=[...e];for(let e=0;e<n.length;e+=1){let t=n[e];if(t===`--port`||t===`-p`){let t=n[e+1],r=Number(t);if(!Number.isNaN(r)&&t)return r;continue}if(t.startsWith(`--port=`)){let e=Number(t.slice(7));if(!Number.isNaN(e))return e;continue}if(t.startsWith(`-p=`)){let e=Number(t.slice(3));if(!Number.isNaN(e))return e}}return Number(t)}async function _r(){let e=qe(process.argv.slice(2)),t=process.env.PLAYWRIGHT_PORT??e.config.commands.tools?.port??e.config.commands.exec?.port??u,n=new I;n.name(`browse-tool`).description(`MCP server for browser automation using Playwright with profile management, page registry, and multi-browser support`).option(`--config <path>`,`Path to browse-tool config file or config directory`).version(ve),n.addCommand(Zn),n.addCommand(tn),n.addCommand(Ne),n.addCommand(ct),n.addCommand($n),n.addCommand(Qn),n.addCommand(lt),n.addCommand(ot),n.addCommand(st);let r=dr();nr(process.argv.slice(2),process.env.PLAYWRIGHT_SKIP_DYNAMIC_TOOL_COMMANDS===`1`)&&await hr(r,{port:gr(process.argv.slice(2),t)}),n.addCommand(r),await n.parseAsync(process.argv)}_r().catch(e=>{console.error(e instanceof Error?e.message:String(e)),process.exit(1)});export{};
431
+ `)})]}),L(`body`,{children:t})]})}function Xt(e){let t=new I;return t.get(`/`,async t=>{try{let n=e.get(r.BrowserService),i=e.get(r.PageRegistry),a=n.listBrowsers(),o=i.list(),s=a.map(e=>{let t=o.filter(t=>t.browserId===e.id);return{id:e.id,profileName:e.profileName,currentPageId:e.currentPageId,createdAt:e.createdAt,pages:t.map(e=>({id:e.id,url:e.url,title:e.title,createdAt:e.createdAt}))}}),c={totalBrowsers:a.length,totalPages:o.length};return t.html(L(Yt,{title:`Playwright MCP Dashboard`,children:L(Jt,{browsers:s,stats:c})}))}catch(e){return console.error(`Failed to render dashboard:`,e),t.html(L(Yt,{title:`Playwright MCP Dashboard - Error`,children:R(`div`,{style:`padding: 2rem; text-align: center;`,children:[L(`h1`,{children:`Failed to load dashboard`}),L(`p`,{children:e instanceof Error?e.message:String(e)}),L(`a`,{href:`/`,style:`color: #2196f3; text-decoration: underline;`,children:`Retry`})]})}),500)}}),t}function Zt(e){if(!e.isError)return;let t=e.content[0];return t?.type===`text`&&typeof t.text==`string`?t.text:`Unknown tool execution error`}function Qt(e){try{return e.get(r.TelemetryService)}catch{return new f}}function $t(e){return e.startsWith(`browser-`)||e===`run-spec`||e===`discover-specs`?e.replaceAll(`-`,`_`):e}function en(e){let t=new I,n=e.get(r.PageRegistry),i=e.get(r.BrowserService),a=e.get(r.ExtensionTaskQueue),o=Qt(e),s=new Pt(n,a,o,i);t.use(`*`,_e({origin:e=>e??null,credentials:!0,allowMethods:[`GET`,`POST`,`DELETE`,`OPTIONS`],allowHeaders:[`Content-Type`,`Accept`,`Authorization`]})),t.get(`/health`,e=>{try{let t=i.listBrowsers(),n={status:`healthy`,service:`browse-tool-http`,serviceName:`browse-tool-http`,timestamp:new Date().toISOString(),browsers:{count:t.length,instances:t.map(e=>({id:e.id,pageCount:e.pageIds.size,createdAt:e.createdAt.toISOString()}))}};return e.json(n)}catch(t){return e.json({status:`unhealthy`,service:`browse-tool-http`,serviceName:`browse-tool-http`,error:t instanceof Error?t.message:String(t)},503)}}),t.post(`/execute`,async t=>{try{let i=await t.req.json();if(!i.tool)return t.json({success:!1,error:`Missing "tool" field in request body`},400);let a=typeof i.arguments?.pageId==`string`?i.arguments.pageId:void 0;return await o.runInSpan(`browse_tool.http.execute`,{attributes:{"http.method":`POST`,"http.route":`/execute`,"browse_tool.tool.name":i.tool,"browse_tool.page.id":a}},async a=>{let s=e.getAll(r.Tool),c=$t(i.tool),l=s.find(e=>e.getDefinition().name===c);if(process.env.BROWSE_TOOL_DEBUG_EXTENSION_RECORDING===`1`&&i.tool===`browser_launch`){let e=typeof i.arguments?.videoDir==`string`?i.arguments.videoDir:typeof i.arguments?.video_dir==`string`?i.arguments.video_dir:``;console.log(`[ExtensionRecordingDebug] http execute browser_launch mode=${String(i.arguments?.mode??``)} videoDir=${e}`)}if(!l)return a?.setStatus({code:P.ERROR,message:`Tool "${i.tool}" not found`}),o.log(`warn`,`browse-tool HTTP execute rejected unknown tool`,{attributes:{"http.route":`/execute`,"browse_tool.tool.name":i.tool}}),t.json({success:!1,error:`Tool "${i.tool}" not found`},404);let u=i.arguments||{},d;try{d=l.getInputSchema().parse(le(u,l.getInputSchema()))}catch(e){if(e instanceof N.ZodError){let n=ue(e,{schemaName:c,schema:l.getInputSchema()});return a?.setStatus({code:P.ERROR,message:n}),t.json({success:!1,result:{content:[{type:`text`,text:n}],isError:!0},error:n})}throw e}let f=await w(t.req.header(T),()=>l.execute(d)),p=d,m=typeof p.pageId==`string`?p.pageId:void 0;if(m){let t=e.get(r.BrowserService),i=n.get(m);i&&t.recordBrowserActivity(i.browserId,m)}let h=Zt(f);return h?(a?.setStatus({code:P.ERROR,message:h}),o.log(`error`,`browse-tool HTTP execute failed`,{attributes:{"http.route":`/execute`,"browse_tool.tool.name":c,"browse_tool.page.id":m}})):o.log(`info`,`browse-tool HTTP execute succeeded`,{attributes:{"http.route":`/execute`,"browse_tool.tool.name":c,"browse_tool.page.id":m}}),t.json({success:!f.isError,result:f,error:h})})}catch(e){return o.log(`error`,`browse-tool HTTP execute request crashed`,{attributes:{"http.route":`/execute`},exception:e}),t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/browsers`,t=>{try{let n=e.get(r.BrowserService).listBrowsers();return t.json({browsers:n.map(e=>({id:e.id,profileName:e.profileName,pageIds:Array.from(e.pageIds),currentPageId:e.currentPageId,createdAt:e.createdAt.toISOString()}))})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/tools`,t=>{try{let n=e.getAll(r.Tool);return t.json({tools:n.map(e=>e.getDefinition())})}catch(e){return t.json({error:e instanceof Error?e.message:String(e)},500)}}),t.get(`/custom-tools`,async e=>{let t=e.req.query(`dir`);if(!t)return e.json({tools:[],error:`Missing "dir" query parameter`},400);try{let n=await s.listTools(t);return e.json({tools:n})}catch(t){return e.json({tools:[],error:t instanceof Error?t.message:String(t)},400)}}),t.post(`/custom-tools`,async t=>{let i=t.req.query(`dir`);if(!i)return t.json({success:!1,error:`Missing "dir" query parameter`},400);try{let a=await t.req.json();if(!a.tool)return t.json({success:!1,error:`Missing "tool" field in request body`},400);let c=typeof a.arguments?.pageId==`string`?a.arguments.pageId:void 0;return await o.runInSpan(`browse_tool.http.custom_tool.execute`,{attributes:{"http.method":`POST`,"http.route":`/custom-tools`,"browse_tool.tool.name":a.tool,"browse_tool.page.id":c,"browse_tool.custom_tools.directory":i}},async l=>{let u=await w(t.req.header(T),()=>s.executeTool(i,a.tool,a.arguments||{}));if(c){let t=e.get(r.BrowserService),i=n.get(c);i&&t.recordBrowserActivity(i.browserId,c)}let d=Zt(u);return d?(l?.setStatus({code:P.ERROR,message:d}),o.log(`error`,`browse-tool custom tool execution failed`,{attributes:{"http.route":`/custom-tools`,"browse_tool.tool.name":a.tool,"browse_tool.page.id":c}})):o.log(`info`,`browse-tool custom tool execution succeeded`,{attributes:{"http.route":`/custom-tools`,"browse_tool.tool.name":a.tool,"browse_tool.page.id":c}}),t.json({success:!u.isError,result:u,error:d})})}catch(e){return o.log(`error`,`browse-tool custom tool request crashed`,{attributes:{"http.route":`/custom-tools`,"browse_tool.custom_tools.directory":i},exception:e}),t.json({success:!1,error:e instanceof Error?e.message:String(e)},500)}});let c=we(e);t.route(`/extension`,c);let l=Ft(e);t.route(`/api`,l);let u=Xt(e);return t.route(`/`,u),t}function tn(e,t,n){e.get(`/ws/extension/:browserId`,n(e=>{let n=e.req.param(`browserId`),i=null;return{onOpen(e,a){if(!n){a.close(1008,`Missing browserId`);return}try{i=t.get(r.WebSocketHub).addConnection(a,n)}catch{a.close(1011,`Internal server error`)}},onMessage(e){if(i)try{let n=t.get(r.WebSocketHub),a=typeof e.data==`string`?e.data:e.data.toString();n.handleMessage(i,a)}catch{}},onClose(){if(i)try{t.get(r.WebSocketHub).removeConnection(i)}catch{}},onError(){if(i)try{t.get(r.WebSocketHub).removeConnection(i)}catch{}}}})),e.get(`/ws/stats`,e=>{try{let n=t.get(r.WebSocketHub).getStats();return e.json({totalConnections:n.totalConnections,browserCount:n.browserCount,connections:n.connections.map(e=>({id:e.id,browserId:e.browserId,connectedAt:e.connectedAt.toISOString()}))})}catch(t){return e.json({error:t instanceof Error?t.message:String(t)},500)}})}const nn=[`pnpm-workspace.yaml`,`nx.json`,`.git`];function rn(e=process.cwd()){let t=A.resolve(e);for(;;){for(let e of nn)if(j(A.join(t,e)))return t;let e=A.dirname(t);if(e===t)return process.cwd();t=e}}const an=new F(`http-serve`).description(`Start HTTP server for browser automation`).option(`-p, --port <port>`,`Port to listen on`,String(u)).option(`--host <host>`,`Host to bind`,y()).option(`--headless`,`Run browsers in headless mode by default`,!0).option(`--idle-timeout <minutes>`,`Idle timeout in minutes before closing browsers`,`30`).option(`--registry-dir <path>`,`Custom registry path or directory for service discovery`).option(`--registry-path <path>`,`Custom registry path or directory for service discovery`).option(`--pids-dir <path>`,`Custom PIDs directory for process tracking (deprecated)`).option(`--profiles-dir <path>`,`Custom profiles directory for browser profiles`).option(`--snippets-dir <path>`,`Directory used by browser_run_code to save and load reusable snippets`).option(`--proxy-config-dir <path>`,`Custom proxy config directory`).action(async function(e){let t=z(`httpServe`),n={port:B(this,`port`,e.port,t.port===void 0?void 0:String(t.port),process.env.PLAYWRIGHT_PORT),headless:B(this,`headless`,e.headless,t.headless),idleTimeout:B(this,`idleTimeout`,e.idleTimeout,t.idleTimeout===void 0?void 0:String(t.idleTimeout),process.env[x]),host:B(this,`host`,e.host,t.host,process.env.PLAYWRIGHT_HOST),registryDir:B(this,`registryDir`,e.registryDir,t.registryDir,process.env.PLAYWRIGHT_REGISTRY_DIR),registryPath:B(this,`registryPath`,e.registryPath,t.registryPath,process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH),pidsDir:B(this,`pidsDir`,e.pidsDir,t.pidsDir,process.env.PLAYWRIGHT_PIDS_DIR),profilesDir:B(this,`profilesDir`,e.profilesDir,t.profilesDir,process.env.PLAYWRIGHT_PROFILES_DIR),snippetsDir:B(this,`snippetsDir`,e.snippetsDir,t.snippetsDir,process.env.BROWSE_TOOL_SNIPPETS_DIR),proxyConfigDir:B(this,`proxyConfigDir`,e.proxyConfigDir,t.proxyConfigDir,process.env[b])},i;try{let e=Number.parseInt(n.port,10),t=`browse-tool-http`,a=process.env.NODE_ENV||`development`,o=rn(process.cwd()),c=n.registryPath||n.registryDir||process.env.PLAYWRIGHT_REGISTRY_PATH;c&&(process.env.PLAYWRIGHT_REGISTRY_DIR=c,process.env.PORT_REGISTRY_PATH=c,process.env.PROCESS_REGISTRY_PATH=ne(c,`processes.json`)),process.env.PLAYWRIGHT_HOST=n.host,process.env.PLAYWRIGHT_PORT=n.port,process.env[x]=n.idleTimeout,n.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=n.pidsDir),n.profilesDir&&(process.env.PLAYWRIGHT_PROFILES_DIR=n.profilesDir),n.snippetsDir&&(process.env.BROWSE_TOOL_SNIPPETS_DIR=A.resolve(n.snippetsDir)),n.proxyConfigDir&&(process.env[b]=A.resolve(n.proxyConfigDir)),console.log(`Starting HTTP server for browser automation...`),console.log(` Port: ${e}`),console.log(` Host: ${n.host}`),console.log(` Headless: ${n.headless}`),console.log(` Idle Timeout: ${n.idleTimeout} minutes`),process.env.BROWSE_TOOL_SNIPPETS_DIR&&console.log(` Snippets Dir: ${process.env.BROWSE_TOOL_SNIPPETS_DIR}`),process.env.BROWSE_TOOL_PROXY_CONFIG_DIR&&console.log(` Proxy Config Dir: ${process.env[b]}`);let l=s(),u=l.get(r.ProcessRegistryService),f=en(l),{injectWebSocket:p,upgradeWebSocket:m}=ve({app:f});tn(f,l,m);let h=l.get(r.WebSocketHub),g=l.get(r.ExtensionTaskQueue),_=l.get(r.BrowserService),v=l.get(r.ExtensionSessionRegistry),y=l.get(r.PageRegistry),ee=l.get(r.IdleCleanupService);ee.start(),h.setEventHandlers({onSessionRegister:(e,t)=>{if(t.type===`session:register`){let n=t.payload.browserId||e.browserId,r=_.getBrowser(n),i,a;if(r&&(r.mode===`extension`||r.mode===`vm`)){i=r.id;let t=y.findByBrowser(i);a=r.currentPageId||t[0]?.id||``,h.reassignConnection(e.id,i),a||(a=y.registerExtensionPage(i),r.pageIds.add(a),r.currentPageId=a),console.log(`[WebSocket] Extension connected to existing browser: ${i}, pageId: ${a}`)}else{i=n,_.registerExtensionBrowserWithId(i),a=y.registerExtensionPage(i);let e=_.getBrowser(i);e&&(e.pageIds.add(a),e.currentPageId=a),console.log(`[WebSocket] Extension connected with new browser: ${i}, pageId: ${a}`)}let o=y.get(a);o&&(o.extensionTabId=t.payload.tabId,o.url=t.payload.url??o.url);let s=v.register({browserId:i,tabId:t.payload.tabId,url:t.payload.url,metadata:{transport:`websocket`}});h.sendSessionAck(e,t.id??``,s.id,s.controlMode),h.broadcastPageCreated(i,a,y.get(a)?.url)}},onTaskResult:(e,t)=>{if(t.type===`task:result`){let e=g.submitResult({taskId:t.payload.taskId,success:t.payload.success,result:t.payload.result,error:t.payload.error});e?.browserId&&_.recordBrowserActivity(e.browserId,e.pageId)}},onTabMapped:(e,t)=>{if(t.type===`tab:mapped`){let e=y.get(t.payload.pageId);if(!e)return;e.extensionTabId=t.payload.tabId,_.recordBrowserActivity(e.browserId,e.id)}},onDisconnect:e=>{e.sessionId&&v.removeSession(e.sessionId),console.log(`[WebSocket] Extension disconnected: ${e.browserId}`)}});let S=new O(process.env.PORT_REGISTRY_PATH);console.log(c?` Registry path: ${c}`:` Registry path: default (~/.port-registry/ports.json)`);let C=await S.reservePort({repositoryPath:o,serviceName:t,serviceType:`tool`,environment:a,preferredPort:e,pid:process.pid,host:n.host,force:!0,portRange:D,metadata:{headless:n.headless,idleTimeout:n.idleTimeout}});if(!C.success||C.port===void 0)throw Error(C.error||`Failed to reserve port in range ${D.min}-${D.max}`);let w=C.port;C.record.metadata={...C.record.metadata,healthCheckUrl:`${d(n.host,w)}/health`};let T=ge({fetch:f.fetch,port:w,hostname:n.host});p(T),i=await te({repositoryPath:o,serviceName:t,serviceType:`service`,environment:a,pid:process.pid,port:w,host:n.host,metadata:{healthCheckUrl:`${d(n.host,w)}/health`,headless:n.headless,idleTimeout:n.idleTimeout,transport:`http`}},u);let E=d(n.host,w);console.log(`HTTP server listening on ${E}`),console.log(` Health check: ${E}/health`),console.log(` Execute tool: POST ${E}/execute`),console.log(`
432
+ Press Ctrl+C to stop the server`);let k=async e=>{console.log(`\n\n${e} received. Shutting down gracefully...`);try{ee.stop(),T.close(),console.log(`Server closed`);try{await l.get(r.BrowserService).closeAll(),console.log(`All browsers closed`)}catch{}try{i&&await i.release({kill:!1}),console.log(`Service deregistered`)}catch(e){console.error(`Process registry cleanup error:`,e)}console.log(`Goodbye!`),process.exit(0)}catch(e){console.error(`Error during shutdown:`,e),process.exit(1)}};process.on(`SIGINT`,()=>k(`SIGINT`)),process.on(`SIGTERM`,()=>k(`SIGTERM`)),await new Promise(()=>{})}catch(e){if(i)try{await i.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}console.error(`Error starting HTTP server:`,e),process.exit(1)}}),on={browser_click:[`input`],browser_fill:[`input`],browser_type:[`input`],browser_select:[`input`],browser_hover:[`input`],browser_drag:[`input`],browser_press_key:[`input`],browser_upload_file:[`input`],browser_navigate:[`navigation`],browser_go_back:[`navigation`],browser_go_forward:[`navigation`],browser_reload:[`navigation`],browser_wait_for:[`navigation`],browser_snapshot:[`snapshot`],browser_screenshot:[`snapshot`],browser_pdf:[`snapshot`],browser_list_pages:[`page`],browser_new_page:[`page`],browser_select_page:[`page`],browser_close_page:[`page`],browser_resize_page:[`page`,`emulation`],browser_handle_dialog:[`dialog`],browser_list_network_requests:[`network`],browser_get_network_request:[`network`],browser_list_console_messages:[`console`],browser_evaluate_script:[`script`],browser_emulate:[`emulation`],browser_expect:[`testing`],browser_start_trace:[`tracing`],browser_stop_trace:[`tracing`],browser_list_profiles:[`profile`],browser_delete_profile:[`profile`],run_spec:[`spec`,`testing`],discover_specs:[`spec`,`testing`],browser_launch:[`browser`],browser_close:[`browser`],browser_run_code:[`code`,`script`]};function sn(e){let t=new Set;for(let[n,r]of Object.entries(on))r.some(t=>e.includes(t))&&t.add(n);return t}const cn=3e3,ln=[`browser_launch`],un=[`browser_new_page`],dn=[`browser_close`],fn=[`browser_close_page`];function pn(e,t){let n=e.replaceAll(`-`,`_`);return t&&(n===`browser_list`||n===`browser_list_session`||n===`browser_list_sessions`)?`browser_list_session`:e.startsWith(`browser-`)||e===`run-spec`||e===`discover-specs`?n:e}function mn(e){if(!e?.content?.[0]?.text)return null;try{let t=JSON.parse(e.content[0].text);return{browserId:t.browserId,pageId:t.pageId,mode:t.mode,url:t.url,closedBrowserId:t.closedBrowserId,closedPageId:t.closedPageId,browserKeptOpen:t.browserKeptOpen}}catch{return null}}async function hn(e){await new Promise(t=>setTimeout(t,e))}async function gn(e){let t=new AbortController,n=setTimeout(()=>t.abort(),cn);try{let n=await fetch(`${e}/tools`,{signal:t.signal});if(!n.ok)throw Error(`HTTP error ${n.status}: ${n.statusText}`);let r=await n.json();if(r.error)throw Error(r.error);if(!Array.isArray(r.tools))throw Error(`Invalid /tools response: missing tools array`);if(r.tools.length===0)throw Error(`HTTP server returned an empty tool list`);return r.tools}finally{clearTimeout(n)}}function _n(e,t){let n=new URL(`/custom-tools`,e);return n.searchParams.set(`dir`,t),n.toString()}async function vn(e,t){let n=new AbortController,r=setTimeout(()=>n.abort(),cn);try{let r=await fetch(_n(e,t),{signal:n.signal});if(!r.ok)throw Error(`HTTP error ${r.status}: ${r.statusText}`);let i=await r.json();if(i.error)throw Error(i.error);if(!Array.isArray(i.tools))throw Error(`Invalid /custom-tools response: missing tools array`);return i.tools}finally{clearTimeout(r)}}async function yn(e){let t;for(let n=1;n<=4;n+=1)try{return await gn(e)}catch(e){t=e instanceof Error?e:Error(String(e)),n<4&&await hn(150*n)}throw Error(`Failed to fetch tools after 4 attempts: ${t?.message??`Unknown error`}`,{cause:t})}async function bn(e,t){let n;for(let r=1;r<=4;r+=1)try{return await vn(e,t)}catch(e){n=e instanceof Error?e:Error(String(e)),r<4&&await hn(150*r)}throw Error(`Failed to fetch custom tools after 4 attempts: ${n?.message??`Unknown error`}`,{cause:n})}function xn(e){if(!e?.tags?.length&&!e?.exclude?.length)return null;let t=e.tags?.length?sn(e.tags):null,n=new Set(e.exclude??[]);if(!t)return n.size>0?n:null;for(let e of n)t.delete(e);return t}function Sn(e){let t=new Map,n=(e,n)=>{let r=t.get(e)??new Set;r.add(n),t.set(e,r)};return e.defaultMode&&n(`browser_launch`,`mode`),e.enforcedProfileName&&n(`browser_launch`,`profileName`),t}function Cn(e,t){let n=t.get(e.name);if(!n?.size)return e;let r=e.inputSchema;if(!r.properties||typeof r.properties!=`object`)return e;let i=Object.fromEntries(Object.entries(r.properties).filter(([e])=>!n.has(e))),a={...r,properties:i};return Array.isArray(r.required)&&(a.required=r.required.filter(e=>typeof e!=`string`||!n.has(e))),{...e,inputSchema:a}}function wn(e){let{httpBaseUrl:t,sessionTracker:n,defaultMode:r,toolFilter:i,customToolsDir:a,enforcedProfileName:o,proxyConfigDir:s,ownerId:c}=e,u=i?.tags?.length?xn(i):null,d=new Set(i?.exclude??[]),f=u!==null||d.size>0,p=Sn({defaultMode:r,enforcedProfileName:o}),m=new de({name:`browse-tool`,version:`0.1.0`},{capabilities:{tools:{}}}),h={name:`browser_list_session`,description:`List all browsers and pages created in this MCP session. Use this to see what resources you have available.`,inputSchema:{type:`object`,properties:{},required:[],additionalProperties:!1}},g=[],_=[],v=new Set;function y(e){return f?e.filter(e=>d.has(e.name)?!1:u===null?!0:u.has(e.name)):e}function b(e){return e.filter(e=>!d.has(e.name))}function ee(e){return o?{...e,profileName:o}:e}function x(e){let t=l({proxy:e.proxy,profileName:typeof e.profileName==`string`?e.profileName:void 0,proxyConfigDir:s});if(!t){let t={...e};return delete t.proxy,t}return{...e,proxy:t}}function S(e){if(!o)return e;let t=e?.content?.[0]?.text;if(!t)return e;try{let n=JSON.parse(t),r=(Array.isArray(n.profiles)?n.profiles:[]).filter(e=>e?.name===o);return{...e,content:[{...e.content[0],text:JSON.stringify({...n,profileCount:r.length,profiles:r},null,2)},...e.content.slice(1)]}}catch{return e}}function C(e){return{name:e.name,description:e.description,inputSchema:e.inputSchema,annotations:e.capabilities}}function w(e){return d.has(e)?!1:u===null?!0:u.has(e)}return m.setRequestHandler(pe,async()=>{try{let e=await yn(t);g=e,v=new Set(e.map(e=>e.name));let r=a?await bn(t,a):[];_=r;let i=[...y(e).map(e=>Cn(e,p)),...b(r).map(e=>C(e))];return n&&i.push(h),{tools:i}}catch(e){if(g.length>0||_.length>0){console.error(`Failed to refresh tools from HTTP server. Returning cached tools:`,e instanceof Error?e.message:String(e));let t=[...y(g).map(e=>Cn(e,p)),...b(_).map(e=>C(e))];return n&&t.push(h),{tools:t}}let r=e instanceof Error?e.message:String(e);throw Error(`Failed to list tools from HTTP server at ${t}: ${r}`,{cause:e})}}),m.setRequestHandler(fe,async e=>{let{name:i,arguments:o}=e.params,s=pn(i,!!n),l=new Set(_.map(e=>e.name));if(s!==`browser_list_session`&&!w(s))return{content:[{type:`text`,text:`Tool "${s}" is not available (filtered by --tags or --exclude)`}],isError:!0};if(s===`browser_list_session`&&n){let e=n.getSessionState();return{content:[{type:`text`,text:JSON.stringify(e,null,2)}]}}let u=o||{};s===`browser_launch`&&r&&u.mode===void 0&&(u={...u,mode:r}),s===`browser_launch`&&(u=ee(u),u=x(u));try{if(a&&!l.has(s)&&!v.has(s))try{_=await bn(t,a),l=new Set(_.map(e=>e.name))}catch(e){console.error(`Failed to refresh custom tools, using cached list: ${e instanceof Error?e.message:String(e)}`)}let e=l.has(s),r=await fetch(e&&a?_n(t,a):`${t}/execute`,{method:`POST`,headers:c?{"Content-Type":`application/json`,[T]:c}:{"Content-Type":`application/json`},body:JSON.stringify({tool:s,arguments:u})});if(!r.ok){let e=await r.text();return{content:[{type:`text`,text:`HTTP error ${r.status}: ${e}`}],isError:!0}}let i=await r.json();if(!i.success)return{content:[{type:`text`,text:i.error||i.result?.content?.[0]?.text||`Unknown error from HTTP server`}],isError:!0};if(n&&i.result){if(ln.includes(s)){let e=mn(i.result);e?.browserId&&(n.trackBrowser(e.browserId,e.mode||`playwright`),e.pageId&&n.trackPage(e.pageId,e.browserId,e.url))}else if(un.includes(s)){let e=mn(i.result);e?.pageId&&e?.browserId&&n.trackPage(e.pageId,e.browserId,e.url)}else if(dn.includes(s)){let e=mn(i.result);e?.closedBrowserId&&n.untrackBrowser(e.closedBrowserId)}else if(fn.includes(s)){let e=mn(i.result);e?.closedPageId&&n.untrackPage(e.closedPageId),e?.browserId&&e.browserKeptOpen===!1&&n.untrackBrowser(e.browserId)}}return s===`browser_list_profiles`&&i.result?S(i.result):i.result||{content:[{type:`text`,text:`No result returned`}]}}catch(e){return{content:[{type:`text`,text:`Failed to execute tool via HTTP server: ${e instanceof Error?e.message:String(e)}`}],isError:!0}}}),m}const Q=`stdio`,Tn=`http`,$=`streamable-http`,En=new Set([Q,Tn,$]),Dn=`chromium`,On=new Set([Dn,`firefox`,`webkit`]),kn=new Set([`playwright`,`extension`,`vm`,`stealth`]),An=`SIGINT`,jn=`SIGTERM`,Mn=`PLAYWRIGHT_PORT`,Nn=[`pnpm-workspace.yaml`,`nx.json`,`.git`],Pn=`browse-tool-mcp-http`,Fn=`service`,In=D.min;var Ln=class extends Error{code=`INVALID_TRANSPORT`;recovery=`Use --type ${[...En].join(`, `)}`;transportType;constructor(e,t){super(`Unknown transport type: ${e}`,t),this.name=`InvalidTransportError`,this.transportType=e}},Rn=class extends Error{code=`INVALID_BROWSER_TYPE`;recovery=`Use --browser ${[...On].join(`, `)}`;browserType;constructor(e,t){super(`Unknown browser type: ${e}`,t),this.name=`InvalidBrowserTypeError`,this.browserType=e}},zn=class extends Error{code=`INVALID_LAUNCH_MODE`;recovery=`Use --mode ${[...kn].join(`, `)}`;launchMode;constructor(e,t){super(`Unknown launch mode: ${e}`,t),this.name=`InvalidLaunchModeError`,this.launchMode=e}};function Bn(e){let t=e.toLowerCase();if(!On.has(t))throw new Rn(e);return t}function Vn(e){let t=e.toLowerCase();if(!kn.has(t))throw new zn(e);return t}function Hn(e){let t=typeof e==`number`?e:Number.parseInt(e,10);if(!Number.isInteger(t)||t<=0||t>65535)throw Error(`Invalid port: ${e}`);return t}function Un(e){return e!=null&&String(e).trim()!==``}function Wn(e){let t=e.toLowerCase();return t===Tn?$:t}function Gn(e,t){let n=e[t];return(Array.isArray(n)?n[0]:n)?.trim()||void 0}function Kn(e=process.cwd()){let t=A.resolve(e);for(;;){for(let e of Nn)if(j(A.join(t,e)))return t;let e=A.dirname(t);if(e===t)return process.cwd();t=e}}function qn(e){return e.split(`,`).map(e=>e.trim()).filter(e=>e.length>0)}function Jn(e){let t=e.tags?qn(e.tags):void 0,n=e.exclude?qn(e.exclude):void 0;if(!(!t?.length&&!n?.length))return{tags:t,exclude:n}}var Yn=class extends Error{code=`BROWSER_CLEANUP_FAILED`;recovery=`Browsers may still be running; check the HTTP server or kill processes manually`;failedBrowserIds;httpBaseUrl;constructor(e,t,n){super(`Failed to close ${e.length} browser(s): ${e.join(`, `)}`,n),this.name=`BrowserCleanupError`,this.failedBrowserIds=e,this.httpBaseUrl=t}},Xn=class extends Error{code=`SESSION_SHUTDOWN_FAILED`;recovery=`Check logs for cleanup and transport errors; processes may need manual cleanup`;signal;constructor(e,t){super(`Shutdown failed for signal ${e}`,t),this.name=`SessionShutdownError`,this.signal=e}},Zn=class extends Error{code=`HTTP_SERVER_START_FAILED`;recovery=`Check if the HTTP server binary exists and the port is available`;constructor(e){super(`Failed to start HTTP server`,e),this.name=`HttpServerStartError`}},Qn=class extends Error{code=`MCP_SERVER_BOOTSTRAP_FAILED`;recovery=`Check if all dependencies are installed and try again`;constructor(e){super(`Failed to start MCP server`,e),this.name=`McpServerBootstrapError`}};async function $n(e,t,n){let r=e.getBrowserIds(),i=[],a;for(let e of r)try{let r=await fetch(`${t}/execute`,{method:`POST`,headers:n?{"Content-Type":`application/json`,[T]:n}:{"Content-Type":`application/json`},body:JSON.stringify({tool:`browser_close`,arguments:{browserId:e}})});if(!r.ok){let e=await r.text().catch(()=>``);throw Error(`HTTP ${r.status} from ${t}/execute: ${e}`)}console.error(` Closed browser: ${e}`)}catch(t){let n=t instanceof Error?t:Error(String(t));console.error(` Failed to close browser: ${e}`,n),i.push(e),a??=n}if(i.length>0)throw new Yn(i,t,{cause:a});e.clear()}async function er(e,t,n,r,i){await e.start();let a=!1,o=async o=>{if(a)return;a=!0,console.error(`\nReceived ${o}, shutting down gracefully...`);let s;try{let e=t.getSessionState();e.totalBrowsers>0&&(console.error(` Cleaning up ${e.totalBrowsers} browser(s) from this session...`),await $n(t,n,r))}catch(e){s=e instanceof Error?e:Error(String(e)),console.error(`Browser cleanup error:`,s)}finally{try{await e.stop()}catch(e){let t=e instanceof Error?e:Error(String(e));console.error(`Transport stop error:`,t),s??=t}try{await i?.()}catch(e){let t=e instanceof Error?e:Error(String(e));console.error(`Process registry cleanup error:`,t),s??=t}if(s){let e=new Xn(o,{cause:s});console.error(`Error during shutdown:`,e),process.exit(1)}process.exit(0)}};process.once(An,()=>o(An)),process.once(jn,()=>o(jn))}const tr=new F(`mcp-serve`).description(`Start Playwright MCP server for browser automation`).option(`-t, --type <type>`,`Transport type: ${[Q,$].join(`, `)}`,Q).option(`-b, --browser <browser>`,`Default browser type: ${[...On].join(`, `)}`,Dn).option(`--headless`,`Run browsers in headless mode by default`,!1).option(`--no-headless`,`Run browsers in headed mode (visible window)`).option(`--host <host>`,`Default host for HTTP services`,y()).option(`--port <port>`,`Port for streamable HTTP transport`,String(In)).option(`--http-port <port>`,`Port for the inner browse-tool HTTP service`).option(`-p, --profile <name>`,`Default profile name for browser sessions`).option(`-m, --mode <mode>`,`Default launch mode: ${[...kn].join(`, `)}`).option(`--tags <tags>`,`Comma-separated tags to filter tools (e.g. input,navigation,snapshot)`).option(`--exclude <tools>`,`Comma-separated tool names to exclude (applied after --tags)`).option(`--custom-tools <path>`,`Path to a folder containing tools.yaml and custom tool scripts`).option(`--chrome-for-testing-path <path>`,`Path to a Chrome for Testing executable for the inner HTTP server`).option(`--snippets-dir <path>`,`Path to a folder where browser_run_code snippets are stored`).option(`--idle-timeout <minutes>`,`Idle timeout in minutes before auto-closing browsers`).option(`--registry-path <path>`,`Custom registry path or directory for service discovery`).option(`--registry-dir <path>`,`Custom registry directory for service discovery`).option(`--pids-dir <path>`,`Custom PIDs directory for process tracking`).option(`--profiles-dir <path>`,`Custom profiles directory for browser profiles`).option(`--proxy-config-dir <path>`,`Custom proxy config directory`).action(async function(e){let t=z(`mcpServe`),n={type:B(this,`type`,e.type,t.type),browser:B(this,`browser`,e.browser,t.browser),headless:B(this,`headless`,e.headless,t.headless),profile:B(this,`profile`,e.profile,t.profile),mode:B(this,`mode`,e.mode,t.mode),host:B(this,`host`,e.host,t.host,process.env.PLAYWRIGHT_HOST),port:B(this,`port`,e.port,t.port,process.env.PLAYWRIGHT_MCP_PORT),httpPort:B(this,`httpPort`,e.httpPort,t.httpPort,process.env[Mn]),tags:B(this,`tags`,e.tags,t.tags),exclude:B(this,`exclude`,e.exclude,t.exclude),customTools:B(this,`customTools`,e.customTools,t.customTools),chromeForTestingPath:B(this,`chromeForTestingPath`,e.chromeForTestingPath,t.chromeForTestingPath,process.env[C]),snippetsDir:B(this,`snippetsDir`,e.snippetsDir,t.snippetsDir),idleTimeout:B(this,`idleTimeout`,e.idleTimeout,t.idleTimeout===void 0?void 0:String(t.idleTimeout),process.env[x]),registryPath:B(this,`registryPath`,e.registryPath,t.registryPath,process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH),registryDir:B(this,`registryDir`,e.registryDir,t.registryDir,process.env.PLAYWRIGHT_REGISTRY_DIR),pidsDir:B(this,`pidsDir`,e.pidsDir,t.pidsDir,process.env.PLAYWRIGHT_PIDS_DIR),profilesDir:B(this,`profilesDir`,e.profilesDir,t.profilesDir,process.env.PLAYWRIGHT_PROFILES_DIR),proxyConfigDir:B(this,`proxyConfigDir`,e.proxyConfigDir,t.proxyConfigDir,process.env[b])},i=Wn(n.type),a=Kn(process.cwd()),o=process.env.NODE_ENV||`development`,s,c;try{if(!En.has(i))throw new Ln(i);let e=Bn(n.browser),t=n.mode?Vn(n.mode):void 0,l=Hn(n.port??In),u=Un(n.httpPort),f=u?Hn(n.httpPort):void 0,p=Jn(n),m=n.customTools?A.resolve(n.customTools):void 0,g=n.chromeForTestingPath?A.resolve(n.chromeForTestingPath):void 0,S=n.snippetsDir?A.resolve(n.snippetsDir):void 0;console.error(`Playwright MCP Server starting...`),console.error(` Transport: ${i}`),i===$&&console.error(` MCP endpoint: http://${n.host??y()}:${l}/mcp`),console.error(` Default browser: ${e}`),console.error(` Headless: ${n.headless}`),t&&console.error(` Default mode: ${t}`),n.profile&&console.error(` Profile: ${n.profile}`),p?.tags?.length&&console.error(` Tags: ${p.tags.join(`, `)}`),p?.exclude?.length&&console.error(` Exclude: ${p.exclude.join(`, `)}`),m&&console.error(` Custom tools: ${m}`),g&&console.error(` Chrome for Testing path: ${g}`),S&&console.error(` Snippets dir: ${S}`),n.idleTimeout&&console.error(` Idle timeout: ${n.idleTimeout} minutes`);let w=n.registryPath||n.registryDir;w&&(process.env.PLAYWRIGHT_REGISTRY_DIR=w,process.env.PORT_REGISTRY_PATH=w,process.env.PROCESS_REGISTRY_PATH=ne(w,`processes.json`),console.error(` Registry path: ${w}`)),n.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=n.pidsDir,console.error(` PIDs dir: ${n.pidsDir}`)),n.profilesDir&&(process.env.PLAYWRIGHT_PROFILES_DIR=n.profilesDir,console.error(` Profiles dir: ${n.profilesDir}`)),n.proxyConfigDir&&(process.env[b]=A.resolve(n.proxyConfigDir),console.error(` Proxy config dir: ${process.env[b]}`)),S&&(process.env.BROWSE_TOOL_SNIPPETS_DIR=S),g&&(process.env[C]=g),n.idleTimeout&&(process.env[x]=n.idleTimeout),n.host&&(process.env.PLAYWRIGHT_HOST=n.host);let T=v(),E=T.get(r.ProcessRegistryService);c=await T.get(r.McpPortAllocationService).acquirePorts({repositoryPath:a,environment:o,host:n.host??y(),pid:process.pid,preferredMcpPort:i===$?l:void 0,preferredHttpPort:f,exactHttpPort:u,reserveMcpPort:i===$,mcpServiceName:Pn,mcpServiceType:`tool`,mcpMetadata:i===$?{healthCheckUrl:`${d(n.host??y(),l)}/health`,transport:$}:void 0});let D=c.httpPort;process.env[Mn]=D.toString();let O=await T.get(r.HttpServerManager).ensureRunning(D,{exactPort:!0});if(!O.running)throw new Zn;let k=d(y(),O.port),re=O.spawned?`spawned`:`reused`;if(console.error(` HTTP server: ${re} on port ${O.port}`),i===Q){let r=new h,i=ce(),l=new _(wn({httpBaseUrl:k,sessionTracker:r,defaultMode:t,toolFilter:p,customToolsDir:m,enforcedProfileName:n.profile,proxyConfigDir:process.env[b],ownerId:i}));s=await te({repositoryPath:a,serviceName:`browse-tool-mcp`,serviceType:Fn,environment:o,pid:process.pid,metadata:{transport:Q,browser:e,mode:t}},E),await er(l,r,k,i,async()=>{await s?.release({kill:!1}),await c?.release()});return}let j=c.mcpPort;if(!j)throw Error(`Failed to reserve ${$} port ${l}`);s=await te({repositoryPath:a,serviceName:Pn,serviceType:Fn,environment:o,pid:process.pid,port:j,host:n.host??y(),metadata:{healthCheckUrl:`${d(n.host??y(),j)}/health`,transport:$}},E);let ie=new ee(({headers:e})=>{let r=Gn(e,`x-profile`)??n.profile,i=new h,a=ce();return{server:wn({httpBaseUrl:k,sessionTracker:i,defaultMode:t,toolFilter:p,customToolsDir:m,enforcedProfileName:r,proxyConfigDir:process.env[b],ownerId:a}),onClose:async()=>{i.getSessionState().totalBrowsers>0&&await $n(i,k,a)}}},{host:n.host??y(),port:j});try{await ie.start()}catch(e){try{await s.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}throw await c.release(),c=void 0,e}let ae=!1,M=async e=>{if(!ae){ae=!0,console.error(`\nReceived ${e}, shutting down gracefully...`);try{await ie.stop();try{await s?.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}await c?.release(),c=void 0,process.exit(0)}catch(e){console.error(`Transport stop error:`,e),process.exit(1)}}};process.once(An,()=>M(An)),process.once(jn,()=>M(jn))}catch(e){if(s)try{await s.release({kill:!1})}catch(e){console.error(`Process registry cleanup error:`,e)}if(c)try{await c.release()}catch(e){console.error(`Port allocation cleanup error:`,e)}(e instanceof Ln||e instanceof Rn||e instanceof zn||e instanceof Zn)&&(console.error(`Error [${e.code}]: ${e.message}`),console.error(`Recovery: ${e.recovery}`),process.exit(1));let t=new Qn({cause:e instanceof Error?e:Error(String(e))});console.error(`Error [${t.code}]: ${t.message}`,t.cause),console.error(`Recovery: ${t.recovery}`),process.exit(1)}}),nr=new F(`status`).description(`Show status of HTTP server and diagnostics`).action(async()=>{try{let e=z(`httpServe`),t=process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH??e.registryPath??e.registryDir;t&&(process.env.PLAYWRIGHT_REGISTRY_DIR=t,process.env.PLAYWRIGHT_REGISTRY_PATH=t,process.env.PORT_REGISTRY_PATH=t),!process.env.PLAYWRIGHT_PIDS_DIR&&e.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=e.pidsDir),console.log(`browse-tool Status
433
+ `),console.log(`-`.repeat(50));let n=await m().get(r.HttpServerManager).getStatus();if(n.running){let t=process.env.PLAYWRIGHT_HOST??e.host??y();console.log(`HTTP Server: Running`),console.log(` PID: ${n.pid}`),console.log(` Port: ${n.port}`),console.log(` Health: ${d(t,n.port)}/health`),n.browserCount!==void 0&&console.log(` Browsers: ${n.browserCount}`)}else console.log(`HTTP Server: Not Running`),n.error&&console.log(` Error: ${n.error}`);console.log(``),console.log(`-`.repeat(50)),process.exit(0)}catch(e){console.error(`Error getting status:`,e),process.exit(1)}}),rr=new F(`stop`).description(`Stop HTTP server and clean up registry/PID files`).action(async()=>{try{let e=z(`httpServe`),t=process.env.PLAYWRIGHT_REGISTRY_PATH??process.env.PORT_REGISTRY_PATH??e.registryPath??e.registryDir;t&&(process.env.PLAYWRIGHT_REGISTRY_DIR=t,process.env.PLAYWRIGHT_REGISTRY_PATH=t,process.env.PORT_REGISTRY_PATH=t),!process.env.PLAYWRIGHT_PIDS_DIR&&e.pidsDir&&(process.env.PLAYWRIGHT_PIDS_DIR=e.pidsDir),console.log(`Stopping browse-tool services...`),await m().get(r.HttpServerManager).stop()?(console.log(`HTTP server stopped`),console.log(`Registry and PID files cleaned up`)):console.log(`No HTTP server was running`),console.log(`Done!`),process.exit(0)}catch(e){console.error(`Error stopping services:`,e),process.exit(1)}}),ir=new Set([`--config`]);function ar(e){for(let t=0;t<e.length;t+=1){let n=e[t];if(ir.has(n)){t+=1;continue}if(![...ir].some(e=>n.startsWith(`${e}=`))&&!n.startsWith(`-`))return n}}function or(e,t=!1){if(t)return!1;let n=ar(e);return n?n===`tools`?!0:n===`help`&&e.includes(`tools`):!1}function sr(e){return e.replace(/_/g,`-`)}function cr(e){return e.replace(/([a-z])([A-Z])/g,`$1-$2`).toLowerCase()}function lr(e){let t=new Set;e.type&&t.add(e.type);for(let n of e.oneOf??[])n.type&&t.add(n.type);for(let n of e.anyOf??[])n.type&&t.add(n.type);return t}function ur(e,t){let n=lr(t);if(n.has(`boolean`)){if(e===`true`||e===`1`)return!0;if(e===`false`||e===`0`)return!1}if(t.type===`number`||t.type===`integer`){let t=Number(e);if(Number.isNaN(t))throw Error(`Invalid number: ${e}`);return t}if(t.type===`boolean`){if(e===`true`||e===`1`)return!0;if(e===`false`||e===`0`)return!1;throw Error(`Invalid boolean: ${e}`)}if(n.has(`array`)||n.has(`object`))try{return JSON.parse(e)}catch{if(t.type===`array`||t.type===`object`)throw Error(`Invalid JSON: ${e}`)}if(n.has(`number`)||n.has(`integer`)){let t=Number(e);if(!Number.isNaN(t)&&e.trim()!==``)return t}return e}function dr(e,t){let n=cr(e),r=lr(t);return t.type===`boolean`?`--${n}`:r.has(`boolean`)&&r.has(`object`)?`--${n} [value]`:`--${n} <value>`}function fr(e){let{definition:t,execute:n}=e,r=new F(sr(t.name));r.description(t.description);let i=t.inputSchema,a=i.properties||{},o=new Set(i.required||[]);for(let[e,t]of Object.entries(a)){let n=t,i=dr(e,n),a=n.description||``;n.enum&&n.enum.length>0&&(a+=` (choices: ${n.enum.join(`, `)})`),n.default!==void 0&&(a+=` (default: ${JSON.stringify(n.default)})`),o.has(e)&&(a+=` [required]`),n.type===`boolean`?n.default===!0?r.option(`--no-${cr(e)}`,`Disable ${a}`):r.option(i,a,n.default):n.enum&&n.enum.length>0?r.addOption(new he(i,a).choices(n.enum)):r.option(i,a)}return r.action(async function(){let e=this.optsWithGlobals(),r=z(`tools`),i=B(this,`format`,e.format??`json`,r.format),s=B(this,`color`,e.color??!0,r.color),c=B(this,`port`,String(e.port??u),r.port===void 0?void 0:String(r.port),process.env.PLAYWRIGHT_PORT),l={format:i,color:s},d=Qe(t.name);try{let r={};for(let[t,n]of Object.entries(a)){let i=n,a=cr(t).replace(/-([a-z])/g,(e,t)=>t.toUpperCase()),o=this.getOptionValueSourceWithGlobals(a),s=e[a];i.type===`boolean`&&i.default===!0&&(s=e[a]),(o===void 0||o===`default`||o===`implied`)&&d[t]!==void 0&&(s=d[t]),s===void 0&&i.default!==void 0&&(s=i.default),s!==void 0&&(typeof s==`string`&&i.type!==`string`?r[t]=ur(s,i):r[t]=s)}for(let e of o)if(r[e]===void 0){let t=cr(e);console.error(U(`Missing required option: --${t}`,l.color)),process.exit(1)}let i=Number.parseInt(c,10),s=V({port:i}),u=n?await n(r,{command:this,formatterOptions:l,port:i}):await s.execute(t.name,r),f=tt(u,l);f&&console.log(f),u.isError&&process.exit(1)}catch(e){console.error(U(e instanceof Error?e:String(e),l.color)),process.exit(1)}}),r}function pr(e){return e.map(fr)}const mr={name:`browser_list_session`,description:`List browsers and pages currently tracked by the browse-tool HTTP server.`,inputSchema:{type:`object`,properties:{},required:[],additionalProperties:!1}};function hr(){return new F(`tools`).description(`Execute browser automation tools`).option(`-f, --format <format>`,`Output format: json, text, quiet`,`json`).option(`--no-color`,`Disable colored output`).option(`-p, --port <port>`,`HTTP server port`,String(u))}function gr(e){return{browserCount:e.length,browsers:e.map(e=>({browserId:e.id,profileName:e.profileName,currentPageId:e.currentPageId,pageIds:e.pageIds,createdAt:e.createdAt}))}}function _r(){return{definition:mr,execute:async(e,t)=>{let n=await V({port:t.port}).listBrowsers();return{content:[{type:`text`,text:JSON.stringify(gr(n),null,2)}]}}}}function vr(e){return[...e.map(e=>({definition:e})),_r()]}async function yr(e,t){let n=z(`tools`),r=Number(t?.port??process.env.PLAYWRIGHT_PORT??n.port??u);try{let t=pr(vr(await V({port:r}).listTools()));for(let n of t)e.addCommand(n)}catch(e){let t=e instanceof Error?e.message:String(e);process.stderr.write(`${U(`Failed to load tool commands: ${t}`,!0)}\n`),process.stderr.write(`Make sure the HTTP server is running: browse-tool http-serve
434
+ `)}}function br(e,t){let n=[...e];for(let e=0;e<n.length;e+=1){let t=n[e];if(t===`--port`||t===`-p`){let t=n[e+1],r=Number(t);if(!Number.isNaN(r)&&t)return r;continue}if(t.startsWith(`--port=`)){let e=Number(t.slice(7));if(!Number.isNaN(e))return e;continue}if(t.startsWith(`-p=`)){let e=Number(t.slice(3));if(!Number.isNaN(e))return e}}return Number(t)}async function xr(){let e=Ze(process.argv.slice(2)),t=process.env.PLAYWRIGHT_PORT??e.config.commands.tools?.port??e.config.commands.exec?.port??u,n=new F;n.name(`browse-tool`).description(`MCP server for browser automation using Playwright with profile management, page registry, and multi-browser support`).option(`--config <path>`,`Path to browse-tool config file or config directory`).version(xe),n.addCommand(tr),n.addCommand(an),n.addCommand(Ie),n.addCommand(ft),n.addCommand(rr),n.addCommand(nr),n.addCommand(pt),n.addCommand(ut),n.addCommand(dt);let r=hr();or(process.argv.slice(2),process.env.PLAYWRIGHT_SKIP_DYNAMIC_TOOL_COMMANDS===`1`)&&await yr(r,{port:br(process.argv.slice(2),t)}),n.addCommand(r),await n.parseAsync(process.argv)}xr().catch(e=>{console.error(e instanceof Error?e.message:String(e)),process.exit(1)});export{};