rae-playwright-mcp 0.0.1 → 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (160) hide show
  1. package/README.md +63 -21
  2. package/cli.js +7 -3
  3. package/index.js +1 -1
  4. package/lib/common/config.js +282 -0
  5. package/lib/common/config.js.map +7 -0
  6. package/lib/common/configLoader.js +345 -0
  7. package/lib/common/configLoader.js.map +7 -0
  8. package/lib/common/esmLoaderHost.js +105 -0
  9. package/lib/common/esmLoaderHost.js.map +7 -0
  10. package/lib/common/expectBundle.js +44 -0
  11. package/lib/common/expectBundle.js.map +7 -0
  12. package/lib/common/expectBundleImpl.js +15781 -0
  13. package/lib/common/expectBundleImpl.js.map +7 -0
  14. package/lib/common/fixtures.js +303 -0
  15. package/lib/common/fixtures.js.map +7 -0
  16. package/lib/common/globals.js +59 -0
  17. package/lib/common/globals.js.map +7 -0
  18. package/lib/common/ipc.js +61 -0
  19. package/lib/common/ipc.js.map +7 -0
  20. package/lib/common/poolBuilder.js +86 -0
  21. package/lib/common/poolBuilder.js.map +7 -0
  22. package/lib/common/process.js +133 -0
  23. package/lib/common/process.js.map +7 -0
  24. package/lib/common/suiteUtils.js +141 -0
  25. package/lib/common/suiteUtils.js.map +7 -0
  26. package/lib/common/test.js +323 -0
  27. package/lib/common/test.js.map +7 -0
  28. package/lib/common/testLoader.js +102 -0
  29. package/lib/common/testLoader.js.map +7 -0
  30. package/lib/common/testType.js +299 -0
  31. package/lib/common/testType.js.map +7 -0
  32. package/lib/common/validators.js +69 -0
  33. package/lib/common/validators.js.map +7 -0
  34. package/lib/mcp/browser/actions.d.js +17 -0
  35. package/lib/mcp/browser/actions.d.js.map +7 -0
  36. package/lib/mcp/browser/browserContextFactory.js +343 -0
  37. package/lib/mcp/browser/browserContextFactory.js.map +7 -0
  38. package/lib/mcp/browser/browserServerBackend.js +78 -0
  39. package/lib/mcp/browser/browserServerBackend.js.map +7 -0
  40. package/lib/mcp/browser/config.js +421 -0
  41. package/lib/mcp/browser/config.js.map +7 -0
  42. package/lib/mcp/browser/context.js +287 -0
  43. package/lib/mcp/browser/context.js.map +7 -0
  44. package/lib/mcp/browser/response.js +353 -0
  45. package/lib/mcp/browser/response.js.map +7 -0
  46. package/lib/mcp/browser/sessionLog.js +161 -0
  47. package/lib/mcp/browser/sessionLog.js.map +7 -0
  48. package/lib/mcp/browser/tab.js +329 -0
  49. package/lib/mcp/browser/tab.js.map +7 -0
  50. package/lib/mcp/browser/tools/common.js +64 -0
  51. package/lib/mcp/browser/tools/common.js.map +7 -0
  52. package/lib/mcp/browser/tools/console.js +45 -0
  53. package/lib/mcp/browser/tools/console.js.map +7 -0
  54. package/lib/mcp/browser/tools/dialogs.js +61 -0
  55. package/lib/mcp/browser/tools/dialogs.js.map +7 -0
  56. package/lib/mcp/browser/tools/evaluate.js +60 -0
  57. package/lib/mcp/browser/tools/evaluate.js.map +7 -0
  58. package/lib/mcp/browser/tools/files.js +59 -0
  59. package/lib/mcp/browser/tools/files.js.map +7 -0
  60. package/lib/mcp/browser/tools/form.js +64 -0
  61. package/lib/mcp/browser/tools/form.js.map +7 -0
  62. package/lib/mcp/browser/tools/install.js +70 -0
  63. package/lib/mcp/browser/tools/install.js.map +7 -0
  64. package/lib/mcp/browser/tools/keyboard.js +85 -0
  65. package/lib/mcp/browser/tools/keyboard.js.map +7 -0
  66. package/lib/mcp/browser/tools/mouse.js +108 -0
  67. package/lib/mcp/browser/tools/mouse.js.map +7 -0
  68. package/lib/mcp/browser/tools/navigate.js +63 -0
  69. package/lib/mcp/browser/tools/navigate.js.map +7 -0
  70. package/lib/mcp/browser/tools/network.js +61 -0
  71. package/lib/mcp/browser/tools/network.js.map +7 -0
  72. package/lib/mcp/browser/tools/pdf.js +49 -0
  73. package/lib/mcp/browser/tools/pdf.js.map +7 -0
  74. package/lib/mcp/browser/tools/runCode.js +78 -0
  75. package/lib/mcp/browser/tools/runCode.js.map +7 -0
  76. package/lib/mcp/browser/tools/screenshot.js +106 -0
  77. package/lib/mcp/browser/tools/screenshot.js.map +7 -0
  78. package/lib/mcp/browser/tools/snapshot.js +192 -0
  79. package/lib/mcp/browser/tools/snapshot.js.map +7 -0
  80. package/lib/mcp/browser/tools/tabs.js +68 -0
  81. package/lib/mcp/browser/tools/tabs.js.map +7 -0
  82. package/lib/mcp/browser/tools/tool.js +51 -0
  83. package/lib/mcp/browser/tools/tool.js.map +7 -0
  84. package/lib/mcp/browser/tools/tracing.js +76 -0
  85. package/lib/mcp/browser/tools/tracing.js.map +7 -0
  86. package/lib/mcp/browser/tools/utils.js +95 -0
  87. package/lib/mcp/browser/tools/utils.js.map +7 -0
  88. package/lib/mcp/browser/tools/verify.js +144 -0
  89. package/lib/mcp/browser/tools/verify.js.map +7 -0
  90. package/lib/mcp/browser/tools/wait.js +64 -0
  91. package/lib/mcp/browser/tools/wait.js.map +7 -0
  92. package/lib/mcp/browser/tools.js +83 -0
  93. package/lib/mcp/browser/tools.js.map +7 -0
  94. package/lib/mcp/browser/watchdog.js +45 -0
  95. package/lib/mcp/browser/watchdog.js.map +7 -0
  96. package/lib/mcp/config.d.js +17 -0
  97. package/lib/mcp/config.d.js.map +7 -0
  98. package/lib/mcp/config.d.ts +197 -0
  99. package/lib/mcp/extension/cdpRelay.js +352 -0
  100. package/lib/mcp/extension/cdpRelay.js.map +7 -0
  101. package/lib/mcp/extension/extensionContextFactory.js +77 -0
  102. package/lib/mcp/extension/extensionContextFactory.js.map +7 -0
  103. package/lib/mcp/extension/protocol.js +29 -0
  104. package/lib/mcp/extension/protocol.js.map +7 -0
  105. package/lib/mcp/index.js +62 -0
  106. package/lib/mcp/index.js.map +7 -0
  107. package/lib/mcp/log.js +36 -0
  108. package/lib/mcp/log.js.map +7 -0
  109. package/lib/mcp/program.js +94 -0
  110. package/lib/mcp/program.js.map +7 -0
  111. package/lib/mcp/sdk/exports.js +29 -0
  112. package/lib/mcp/sdk/exports.js.map +7 -0
  113. package/lib/mcp/sdk/http.js +153 -0
  114. package/lib/mcp/sdk/http.js.map +7 -0
  115. package/lib/mcp/sdk/inProcessTransport.js +72 -0
  116. package/lib/mcp/sdk/inProcessTransport.js.map +7 -0
  117. package/lib/mcp/sdk/server.js +208 -0
  118. package/lib/mcp/sdk/server.js.map +7 -0
  119. package/lib/mcp/sdk/tool.js +48 -0
  120. package/lib/mcp/sdk/tool.js.map +7 -0
  121. package/lib/mcp/test/browserBackend.js +99 -0
  122. package/lib/mcp/test/browserBackend.js.map +7 -0
  123. package/lib/mcp/test/generatorTools.js +123 -0
  124. package/lib/mcp/test/generatorTools.js.map +7 -0
  125. package/lib/mcp/test/plannerTools.js +145 -0
  126. package/lib/mcp/test/plannerTools.js.map +7 -0
  127. package/lib/mcp/test/seed.js +83 -0
  128. package/lib/mcp/test/seed.js.map +7 -0
  129. package/lib/mcp/test/streams.js +45 -0
  130. package/lib/mcp/test/streams.js.map +7 -0
  131. package/lib/mcp/test/testBackend.js +100 -0
  132. package/lib/mcp/test/testBackend.js.map +7 -0
  133. package/lib/mcp/test/testContext.js +280 -0
  134. package/lib/mcp/test/testContext.js.map +7 -0
  135. package/lib/mcp/test/testTool.js +31 -0
  136. package/lib/mcp/test/testTool.js.map +7 -0
  137. package/lib/mcp/test/testTools.js +109 -0
  138. package/lib/mcp/test/testTools.js.map +7 -0
  139. package/lib/third_party/pirates.js +63 -0
  140. package/lib/third_party/pirates.js.map +7 -0
  141. package/lib/third_party/tsconfig-loader.js +104 -0
  142. package/lib/third_party/tsconfig-loader.js.map +7 -0
  143. package/lib/transform/babelBundle.js +44 -0
  144. package/lib/transform/babelBundle.js.map +7 -0
  145. package/lib/transform/babelBundleImpl.js +68726 -0
  146. package/lib/transform/babelBundleImpl.js.map +7 -0
  147. package/lib/transform/babelHighlightUtils.js +64 -0
  148. package/lib/transform/babelHighlightUtils.js.map +7 -0
  149. package/lib/transform/compilationCache.js +273 -0
  150. package/lib/transform/compilationCache.js.map +7 -0
  151. package/lib/transform/esmLoader.js +104 -0
  152. package/lib/transform/esmLoader.js.map +7 -0
  153. package/lib/transform/portTransport.js +68 -0
  154. package/lib/transform/portTransport.js.map +7 -0
  155. package/lib/transform/transform.js +297 -0
  156. package/lib/transform/transform.js.map +7 -0
  157. package/lib/util.js +404 -0
  158. package/lib/utilsBundle.js +44 -0
  159. package/lib/utilsBundleImpl.js +13122 -0
  160. package/package.json +4 -3
@@ -0,0 +1,343 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+ var browserContextFactory_exports = {};
30
+ __export(browserContextFactory_exports, {
31
+ SharedContextFactory: () => SharedContextFactory,
32
+ contextFactory: () => contextFactory,
33
+ identityBrowserContextFactory: () => identityBrowserContextFactory
34
+ });
35
+ module.exports = __toCommonJS(browserContextFactory_exports);
36
+ var import_crypto = __toESM(require("crypto"));
37
+ var import_fs = __toESM(require("fs"));
38
+ var import_net = __toESM(require("net"));
39
+ var import_os = __toESM(require("os"));
40
+ var import_path = __toESM(require("path"));
41
+ var playwright = __toESM(require("playwright-core"));
42
+ var import_registry = require("playwright-core/lib/server/registry/index");
43
+ var import_server = require("playwright-core/lib/server");
44
+ var import_log = require("../log");
45
+ var import_config = require("./config");
46
+ var import_server2 = require("../sdk/server");
47
+ function contextFactory(config) {
48
+ if (config.sharedBrowserContext)
49
+ return SharedContextFactory.create(config);
50
+ if (config.browser.remoteEndpoint)
51
+ return new RemoteContextFactory(config);
52
+ if (config.browser.cdpEndpoint)
53
+ return new CdpContextFactory(config);
54
+ if (config.browser.isolated)
55
+ return new IsolatedContextFactory(config);
56
+ return new PersistentContextFactory(config);
57
+ }
58
+ function identityBrowserContextFactory(browserContext) {
59
+ return {
60
+ createContext: async (clientInfo, abortSignal, toolName) => {
61
+ return {
62
+ browserContext,
63
+ close: async () => {
64
+ }
65
+ };
66
+ }
67
+ };
68
+ }
69
+ class BaseContextFactory {
70
+ constructor(name, config) {
71
+ this._logName = name;
72
+ this.config = config;
73
+ }
74
+ async _obtainBrowser(clientInfo) {
75
+ if (this._browserPromise)
76
+ return this._browserPromise;
77
+ (0, import_log.testDebug)(`obtain browser (${this._logName})`);
78
+ this._browserPromise = this._doObtainBrowser(clientInfo);
79
+ void this._browserPromise.then((browser) => {
80
+ browser.on("disconnected", () => {
81
+ this._browserPromise = void 0;
82
+ });
83
+ }).catch(() => {
84
+ this._browserPromise = void 0;
85
+ });
86
+ return this._browserPromise;
87
+ }
88
+ async _doObtainBrowser(clientInfo) {
89
+ throw new Error("Not implemented");
90
+ }
91
+ async createContext(clientInfo) {
92
+ (0, import_log.testDebug)(`create browser context (${this._logName})`);
93
+ const browser = await this._obtainBrowser(clientInfo);
94
+ const harDir = await computeHarDir(clientInfo, this.config.runId);
95
+ const browserContext = await this._doCreateContext(browser, harDir);
96
+ await addInitScript(browserContext, this.config.browser.initScript);
97
+ return {
98
+ browserContext,
99
+ close: (afterClose) => this._closeBrowserContext(browserContext, browser, afterClose)
100
+ };
101
+ }
102
+ async _doCreateContext(browser, harDir) {
103
+ throw new Error("Not implemented");
104
+ }
105
+ async _closeBrowserContext(browserContext, browser, afterClose) {
106
+ (0, import_log.testDebug)(`close browser context (${this._logName})`);
107
+ if (browser.contexts().length === 1)
108
+ this._browserPromise = void 0;
109
+ await browserContext.close().catch(import_log.logUnhandledError);
110
+ await afterClose();
111
+ if (browser.contexts().length === 0) {
112
+ (0, import_log.testDebug)(`close browser (${this._logName})`);
113
+ await browser.close().catch(import_log.logUnhandledError);
114
+ }
115
+ }
116
+ }
117
+ class IsolatedContextFactory extends BaseContextFactory {
118
+ constructor(config) {
119
+ super("isolated", config);
120
+ }
121
+ async _doObtainBrowser(clientInfo) {
122
+ await injectCdpPort(this.config.browser);
123
+ const browserType = playwright[this.config.browser.browserName];
124
+ const tracesDir = await computeTracesDir(this.config, clientInfo);
125
+ if (tracesDir && this.config.saveTrace)
126
+ await startTraceServer(this.config, tracesDir);
127
+ return browserType.launch({
128
+ tracesDir,
129
+ ...this.config.browser.launchOptions,
130
+ handleSIGINT: false,
131
+ handleSIGTERM: false
132
+ }).catch((error) => {
133
+ if (error.message.includes("Executable doesn't exist"))
134
+ throw new Error(`Browser specified in your config is not installed. Either install it (likely) or change the config.`);
135
+ throw error;
136
+ });
137
+ }
138
+ async _doCreateContext(browser, harDir) {
139
+ return browser.newContext(browserContextOptionsFromConfig(this.config, harDir));
140
+ }
141
+ }
142
+ class CdpContextFactory extends BaseContextFactory {
143
+ constructor(config) {
144
+ super("cdp", config);
145
+ }
146
+ async _doObtainBrowser() {
147
+ return playwright.chromium.connectOverCDP(this.config.browser.cdpEndpoint, { headers: this.config.browser.cdpHeaders });
148
+ }
149
+ async _doCreateContext(browser, harDir) {
150
+ if (this.config.browser.isolated) {
151
+ return await browser.newContext(browserContextOptionsFromConfig(this.config, harDir));
152
+ }
153
+ return browser.contexts()[0];
154
+ }
155
+ }
156
+ class RemoteContextFactory extends BaseContextFactory {
157
+ constructor(config) {
158
+ super("remote", config);
159
+ }
160
+ async _doObtainBrowser() {
161
+ const url = new URL(this.config.browser.remoteEndpoint);
162
+ url.searchParams.set("browser", this.config.browser.browserName);
163
+ if (this.config.browser.launchOptions)
164
+ url.searchParams.set("launch-options", JSON.stringify(this.config.browser.launchOptions));
165
+ return playwright[this.config.browser.browserName].connect(String(url));
166
+ }
167
+ async _doCreateContext(browser, harDir) {
168
+ return browser.newContext(browserContextOptionsFromConfig(this.config, harDir));
169
+ }
170
+ }
171
+ class PersistentContextFactory {
172
+ constructor(config) {
173
+ this.name = "persistent";
174
+ this.description = "Create a new persistent browser context";
175
+ this._userDataDirs = /* @__PURE__ */ new Set();
176
+ this.config = config;
177
+ }
178
+ async createContext(clientInfo) {
179
+ await injectCdpPort(this.config.browser);
180
+ (0, import_log.testDebug)("create browser context (persistent)");
181
+ const userDataDir = this.config.browser.userDataDir ?? await this._createUserDataDir(clientInfo);
182
+ const tracesDir = await computeTracesDir(this.config, clientInfo);
183
+ if (tracesDir && this.config.saveTrace)
184
+ await startTraceServer(this.config, tracesDir);
185
+ const harDir = await computeHarDir(clientInfo, this.config.runId);
186
+ this._userDataDirs.add(userDataDir);
187
+ (0, import_log.testDebug)("lock user data dir", userDataDir);
188
+ const browserType = playwright[this.config.browser.browserName];
189
+ for (let i = 0; i < 5; i++) {
190
+ const launchOptions = {
191
+ tracesDir,
192
+ ...this.config.browser.launchOptions,
193
+ ...browserContextOptionsFromConfig(this.config, harDir),
194
+ handleSIGINT: false,
195
+ handleSIGTERM: false,
196
+ ignoreDefaultArgs: [
197
+ "--disable-extensions"
198
+ ],
199
+ assistantMode: true
200
+ };
201
+ try {
202
+ const browserContext = await browserType.launchPersistentContext(userDataDir, launchOptions);
203
+ await addInitScript(browserContext, this.config.browser.initScript);
204
+ const close = (afterClose) => this._closeBrowserContext(browserContext, userDataDir, afterClose);
205
+ return { browserContext, close };
206
+ } catch (error) {
207
+ if (error.message.includes("Executable doesn't exist"))
208
+ throw new Error(`Browser specified in your config is not installed. Either install it (likely) or change the config.`);
209
+ if (error.message.includes("ProcessSingleton") || error.message.includes("Invalid URL")) {
210
+ await new Promise((resolve) => setTimeout(resolve, 1e3));
211
+ continue;
212
+ }
213
+ throw error;
214
+ }
215
+ }
216
+ throw new Error(`Browser is already in use for ${userDataDir}, use --isolated to run multiple instances of the same browser`);
217
+ }
218
+ async _closeBrowserContext(browserContext, userDataDir, afterClose) {
219
+ (0, import_log.testDebug)("close browser context (persistent)");
220
+ (0, import_log.testDebug)("release user data dir", userDataDir);
221
+ await browserContext.close().catch(() => {
222
+ });
223
+ await afterClose();
224
+ this._userDataDirs.delete(userDataDir);
225
+ if (process.env.PWMCP_PROFILES_DIR_FOR_TEST && userDataDir.startsWith(process.env.PWMCP_PROFILES_DIR_FOR_TEST))
226
+ await import_fs.default.promises.rm(userDataDir, { recursive: true }).catch(import_log.logUnhandledError);
227
+ (0, import_log.testDebug)("close browser context complete (persistent)");
228
+ }
229
+ async _createUserDataDir(clientInfo) {
230
+ const dir = process.env.PWMCP_PROFILES_DIR_FOR_TEST ?? import_registry.registryDirectory;
231
+ const browserToken = this.config.browser.launchOptions?.channel ?? this.config.browser?.browserName;
232
+ const rootPath = (0, import_server2.firstRootPath)(clientInfo);
233
+ const rootPathToken = rootPath ? `-${createHash(rootPath)}` : "";
234
+ const result = import_path.default.join(dir, `mcp-${browserToken}${rootPathToken}`);
235
+ await import_fs.default.promises.mkdir(result, { recursive: true });
236
+ return result;
237
+ }
238
+ }
239
+ async function injectCdpPort(browserConfig) {
240
+ if (browserConfig.browserName === "chromium")
241
+ browserConfig.launchOptions.cdpPort = await findFreePort();
242
+ }
243
+ async function findFreePort() {
244
+ return new Promise((resolve, reject) => {
245
+ const server = import_net.default.createServer();
246
+ server.listen(0, () => {
247
+ const { port } = server.address();
248
+ server.close(() => resolve(port));
249
+ });
250
+ server.on("error", reject);
251
+ });
252
+ }
253
+ async function startTraceServer(config, tracesDir) {
254
+ if (!config.saveTrace)
255
+ return;
256
+ const server = await (0, import_server.startTraceViewerServer)();
257
+ const urlPrefix = server.urlPrefix("human-readable");
258
+ const url = urlPrefix + "/trace/index.html?trace=" + tracesDir + "/trace.json";
259
+ console.error("\nTrace viewer listening on " + url);
260
+ }
261
+ function createHash(data) {
262
+ return import_crypto.default.createHash("sha256").update(data).digest("hex").slice(0, 7);
263
+ }
264
+ async function addInitScript(browserContext, initScript) {
265
+ for (const scriptPath of initScript ?? [])
266
+ await browserContext.addInitScript({ path: import_path.default.resolve(scriptPath) });
267
+ }
268
+ class SharedContextFactory {
269
+ static create(config) {
270
+ if (SharedContextFactory._instance)
271
+ throw new Error("SharedContextFactory already exists");
272
+ const baseConfig = { ...config, sharedBrowserContext: false };
273
+ const baseFactory = contextFactory(baseConfig);
274
+ SharedContextFactory._instance = new SharedContextFactory(baseFactory);
275
+ return SharedContextFactory._instance;
276
+ }
277
+ constructor(baseFactory) {
278
+ this._baseFactory = baseFactory;
279
+ }
280
+ async createContext(clientInfo, abortSignal, toolName) {
281
+ if (!this._contextPromise) {
282
+ (0, import_log.testDebug)("create shared browser context");
283
+ this._contextPromise = this._baseFactory.createContext(clientInfo, abortSignal, toolName);
284
+ }
285
+ const { browserContext } = await this._contextPromise;
286
+ (0, import_log.testDebug)(`shared context client connected`);
287
+ return {
288
+ browserContext,
289
+ close: async () => {
290
+ (0, import_log.testDebug)(`shared context client disconnected`);
291
+ }
292
+ };
293
+ }
294
+ static async dispose() {
295
+ await SharedContextFactory._instance?._dispose();
296
+ }
297
+ async _dispose() {
298
+ const contextPromise = this._contextPromise;
299
+ this._contextPromise = void 0;
300
+ if (!contextPromise)
301
+ return;
302
+ const { close } = await contextPromise;
303
+ await close(async () => {
304
+ });
305
+ }
306
+ }
307
+ async function computeTracesDir(config, clientInfo) {
308
+ if (!config.saveTrace && !config.capabilities?.includes("tracing"))
309
+ return;
310
+ return await (0, import_config.outputFile)(config, clientInfo, `traces`, { origin: "code", reason: "Collecting trace" });
311
+ }
312
+ async function computeHarDir(clientInfo, runId) {
313
+ const finalRunId = runId || import_crypto.default.randomUUID();
314
+ const harDir = import_path.default.join(import_os.default.homedir(), ".reverse-api-engineer", "runs", "testing-playwright", finalRunId);
315
+ await import_fs.default.promises.mkdir(harDir, { recursive: true });
316
+ const harPath = import_path.default.join(harDir, "network.har");
317
+ console.error(`HAR recording enabled. HAR file will be saved to: ${harPath}`);
318
+ return harDir;
319
+ }
320
+ function browserContextOptionsFromConfig(config, harDir) {
321
+ const result = { ...config.browser.contextOptions };
322
+ if (config.saveVideo) {
323
+ result.recordVideo = {
324
+ dir: (0, import_config.tmpDir)(),
325
+ size: config.saveVideo
326
+ };
327
+ }
328
+ if (harDir) {
329
+ result.recordHar = {
330
+ path: import_path.default.join(harDir, "network.har"),
331
+ content: "attach",
332
+ mode: "full"
333
+ };
334
+ }
335
+ return result;
336
+ }
337
+ // Annotate the CommonJS export names for ESM import in node:
338
+ 0 && (module.exports = {
339
+ SharedContextFactory,
340
+ contextFactory,
341
+ identityBrowserContextFactory
342
+ });
343
+ //# sourceMappingURL=browserContextFactory.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/mcp/browser/browserContextFactory.ts"],
4
+ "sourcesContent": ["/**\n * Copyright (c) Microsoft Corporation.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport crypto from 'crypto';\nimport fs from 'fs';\nimport net from 'net';\nimport os from 'os';\nimport path from 'path';\n\nimport * as playwright from 'playwright-core';\nimport { registryDirectory } from 'playwright-core/lib/server/registry/index';\nimport { startTraceViewerServer } from 'playwright-core/lib/server';\nimport { logUnhandledError, testDebug } from '../log';\nimport { outputFile, tmpDir } from './config';\nimport { firstRootPath } from '../sdk/server';\n\nimport type { FullConfig } from './config';\nimport type { LaunchOptions, BrowserContextOptions } from '../../../../playwright-core/src/client/types';\nimport type { ClientInfo } from '../sdk/server';\n\nexport function contextFactory(config: FullConfig): BrowserContextFactory {\n if (config.sharedBrowserContext)\n return SharedContextFactory.create(config);\n if (config.browser.remoteEndpoint)\n return new RemoteContextFactory(config);\n if (config.browser.cdpEndpoint)\n return new CdpContextFactory(config);\n if (config.browser.isolated)\n return new IsolatedContextFactory(config);\n return new PersistentContextFactory(config);\n}\n\nexport type BrowserContextFactoryResult = {\n browserContext: playwright.BrowserContext;\n close: (afterClose: () => Promise<void>) => Promise<void>;\n};\n\nexport interface BrowserContextFactory {\n createContext(clientInfo: ClientInfo, abortSignal: AbortSignal, toolName: string | undefined): Promise<BrowserContextFactoryResult>;\n}\n\nexport function identityBrowserContextFactory(browserContext: playwright.BrowserContext): BrowserContextFactory {\n return {\n createContext: async (clientInfo: ClientInfo, abortSignal: AbortSignal, toolName: string | undefined) => {\n return {\n browserContext,\n close: async () => { }\n };\n }\n };\n}\n\nclass BaseContextFactory implements BrowserContextFactory {\n readonly config: FullConfig;\n private _logName: string;\n protected _browserPromise: Promise<playwright.Browser> | undefined;\n\n constructor(name: string, config: FullConfig) {\n this._logName = name;\n this.config = config;\n }\n\n protected async _obtainBrowser(clientInfo: ClientInfo): Promise<playwright.Browser> {\n if (this._browserPromise)\n return this._browserPromise;\n testDebug(`obtain browser (${this._logName})`);\n this._browserPromise = this._doObtainBrowser(clientInfo);\n void this._browserPromise.then(browser => {\n browser.on('disconnected', () => {\n this._browserPromise = undefined;\n });\n }).catch(() => {\n this._browserPromise = undefined;\n });\n return this._browserPromise;\n }\n\n protected async _doObtainBrowser(clientInfo: ClientInfo): Promise<playwright.Browser> {\n throw new Error('Not implemented');\n }\n\n async createContext(clientInfo: ClientInfo): Promise<BrowserContextFactoryResult> {\n testDebug(`create browser context (${this._logName})`);\n const browser = await this._obtainBrowser(clientInfo);\n const harDir = await computeHarDir(clientInfo, this.config.runId);\n const browserContext = await this._doCreateContext(browser, harDir);\n await addInitScript(browserContext, this.config.browser.initScript);\n return {\n browserContext,\n close: (afterClose: () => Promise<void>) => this._closeBrowserContext(browserContext, browser, afterClose)\n };\n }\n\n protected async _doCreateContext(browser: playwright.Browser, harDir: string): Promise<playwright.BrowserContext> {\n throw new Error('Not implemented');\n }\n\n private async _closeBrowserContext(browserContext: playwright.BrowserContext, browser: playwright.Browser, afterClose: () => Promise<void>) {\n testDebug(`close browser context (${this._logName})`);\n if (browser.contexts().length === 1)\n this._browserPromise = undefined;\n await browserContext.close().catch(logUnhandledError);\n await afterClose();\n if (browser.contexts().length === 0) {\n testDebug(`close browser (${this._logName})`);\n await browser.close().catch(logUnhandledError);\n }\n }\n}\n\nclass IsolatedContextFactory extends BaseContextFactory {\n constructor(config: FullConfig) {\n super('isolated', config);\n }\n\n protected override async _doObtainBrowser(clientInfo: ClientInfo): Promise<playwright.Browser> {\n await injectCdpPort(this.config.browser);\n const browserType = playwright[this.config.browser.browserName];\n const tracesDir = await computeTracesDir(this.config, clientInfo);\n if (tracesDir && this.config.saveTrace)\n await startTraceServer(this.config, tracesDir);\n return browserType.launch({\n tracesDir,\n ...this.config.browser.launchOptions,\n handleSIGINT: false,\n handleSIGTERM: false,\n }).catch(error => {\n if (error.message.includes('Executable doesn\\'t exist'))\n throw new Error(`Browser specified in your config is not installed. Either install it (likely) or change the config.`);\n throw error;\n });\n }\n\n protected override async _doCreateContext(browser: playwright.Browser, harDir: string): Promise<playwright.BrowserContext> {\n return browser.newContext(browserContextOptionsFromConfig(this.config, harDir));\n }\n}\n\nclass CdpContextFactory extends BaseContextFactory {\n constructor(config: FullConfig) {\n super('cdp', config);\n }\n\n protected override async _doObtainBrowser(): Promise<playwright.Browser> {\n return playwright.chromium.connectOverCDP(this.config.browser.cdpEndpoint!, { headers: this.config.browser.cdpHeaders });\n }\n\n protected override async _doCreateContext(browser: playwright.Browser, harDir: string): Promise<playwright.BrowserContext> {\n if (this.config.browser.isolated) {\n return await browser.newContext(browserContextOptionsFromConfig(this.config, harDir));\n }\n return browser.contexts()[0];\n }\n}\n\nclass RemoteContextFactory extends BaseContextFactory {\n constructor(config: FullConfig) {\n super('remote', config);\n }\n\n protected override async _doObtainBrowser(): Promise<playwright.Browser> {\n const url = new URL(this.config.browser.remoteEndpoint!);\n url.searchParams.set('browser', this.config.browser.browserName);\n if (this.config.browser.launchOptions)\n url.searchParams.set('launch-options', JSON.stringify(this.config.browser.launchOptions));\n return playwright[this.config.browser.browserName].connect(String(url));\n }\n\n protected override async _doCreateContext(browser: playwright.Browser, harDir: string): Promise<playwright.BrowserContext> {\n return browser.newContext(browserContextOptionsFromConfig(this.config, harDir));\n }\n}\n\nclass PersistentContextFactory implements BrowserContextFactory {\n readonly config: FullConfig;\n readonly name = 'persistent';\n readonly description = 'Create a new persistent browser context';\n\n private _userDataDirs = new Set<string>();\n\n constructor(config: FullConfig) {\n this.config = config;\n }\n\n async createContext(clientInfo: ClientInfo): Promise<BrowserContextFactoryResult> {\n await injectCdpPort(this.config.browser);\n testDebug('create browser context (persistent)');\n const userDataDir = this.config.browser.userDataDir ?? await this._createUserDataDir(clientInfo);\n const tracesDir = await computeTracesDir(this.config, clientInfo);\n if (tracesDir && this.config.saveTrace)\n await startTraceServer(this.config, tracesDir);\n const harDir = await computeHarDir(clientInfo, this.config.runId);\n\n this._userDataDirs.add(userDataDir);\n testDebug('lock user data dir', userDataDir);\n\n const browserType = playwright[this.config.browser.browserName];\n for (let i = 0; i < 5; i++) {\n const launchOptions: LaunchOptions & BrowserContextOptions = {\n tracesDir,\n ...this.config.browser.launchOptions,\n ...browserContextOptionsFromConfig(this.config, harDir),\n handleSIGINT: false,\n handleSIGTERM: false,\n ignoreDefaultArgs: [\n '--disable-extensions',\n ],\n assistantMode: true,\n };\n try {\n const browserContext = await browserType.launchPersistentContext(userDataDir, launchOptions);\n await addInitScript(browserContext, this.config.browser.initScript);\n const close = (afterClose: () => Promise<void>) => this._closeBrowserContext(browserContext, userDataDir, afterClose);\n return { browserContext, close };\n } catch (error: any) {\n if (error.message.includes('Executable doesn\\'t exist'))\n throw new Error(`Browser specified in your config is not installed. Either install it (likely) or change the config.`);\n if (error.message.includes('ProcessSingleton') || error.message.includes('Invalid URL')) {\n // User data directory is already in use, try again.\n await new Promise(resolve => setTimeout(resolve, 1000));\n continue;\n }\n throw error;\n }\n }\n throw new Error(`Browser is already in use for ${userDataDir}, use --isolated to run multiple instances of the same browser`);\n }\n\n private async _closeBrowserContext(browserContext: playwright.BrowserContext, userDataDir: string, afterClose: () => Promise<void>) {\n testDebug('close browser context (persistent)');\n testDebug('release user data dir', userDataDir);\n await browserContext.close().catch(() => { });\n await afterClose();\n this._userDataDirs.delete(userDataDir);\n if (process.env.PWMCP_PROFILES_DIR_FOR_TEST && userDataDir.startsWith(process.env.PWMCP_PROFILES_DIR_FOR_TEST))\n await fs.promises.rm(userDataDir, { recursive: true }).catch(logUnhandledError);\n testDebug('close browser context complete (persistent)');\n }\n\n private async _createUserDataDir(clientInfo: ClientInfo) {\n const dir = process.env.PWMCP_PROFILES_DIR_FOR_TEST ?? registryDirectory;\n const browserToken = this.config.browser.launchOptions?.channel ?? this.config.browser?.browserName;\n // Hesitant putting hundreds of files into the user's workspace, so using it for hashing instead.\n const rootPath = firstRootPath(clientInfo);\n const rootPathToken = rootPath ? `-${createHash(rootPath)}` : '';\n const result = path.join(dir, `mcp-${browserToken}${rootPathToken}`);\n await fs.promises.mkdir(result, { recursive: true });\n return result;\n }\n}\n\nasync function injectCdpPort(browserConfig: FullConfig['browser']) {\n if (browserConfig.browserName === 'chromium')\n (browserConfig.launchOptions as any).cdpPort = await findFreePort();\n}\n\nasync function findFreePort(): Promise<number> {\n return new Promise((resolve, reject) => {\n const server = net.createServer();\n server.listen(0, () => {\n const { port } = server.address() as net.AddressInfo;\n server.close(() => resolve(port));\n });\n server.on('error', reject);\n });\n}\n\nasync function startTraceServer(config: FullConfig, tracesDir: string): Promise<string | undefined> {\n if (!config.saveTrace)\n return;\n\n const server = await startTraceViewerServer();\n const urlPrefix = server.urlPrefix('human-readable');\n const url = urlPrefix + '/trace/index.html?trace=' + tracesDir + '/trace.json';\n // eslint-disable-next-line no-console\n console.error('\\nTrace viewer listening on ' + url);\n}\n\nfunction createHash(data: string): string {\n return crypto.createHash('sha256').update(data).digest('hex').slice(0, 7);\n}\n\nasync function addInitScript(browserContext: playwright.BrowserContext, initScript: string[] | undefined) {\n for (const scriptPath of initScript ?? [])\n await browserContext.addInitScript({ path: path.resolve(scriptPath) });\n}\n\nexport class SharedContextFactory implements BrowserContextFactory {\n private _contextPromise: Promise<BrowserContextFactoryResult> | undefined;\n private _baseFactory: BrowserContextFactory;\n private static _instance: SharedContextFactory | undefined;\n\n static create(config: FullConfig) {\n if (SharedContextFactory._instance)\n throw new Error('SharedContextFactory already exists');\n const baseConfig = { ...config, sharedBrowserContext: false };\n const baseFactory = contextFactory(baseConfig);\n SharedContextFactory._instance = new SharedContextFactory(baseFactory);\n return SharedContextFactory._instance;\n }\n\n private constructor(baseFactory: BrowserContextFactory) {\n this._baseFactory = baseFactory;\n }\n\n async createContext(clientInfo: ClientInfo, abortSignal: AbortSignal, toolName: string | undefined): Promise<{ browserContext: playwright.BrowserContext, close: () => Promise<void> }> {\n if (!this._contextPromise) {\n testDebug('create shared browser context');\n this._contextPromise = this._baseFactory.createContext(clientInfo, abortSignal, toolName);\n }\n\n const { browserContext } = await this._contextPromise;\n testDebug(`shared context client connected`);\n return {\n browserContext,\n close: async () => {\n testDebug(`shared context client disconnected`);\n },\n };\n }\n\n static async dispose() {\n await SharedContextFactory._instance?._dispose();\n }\n\n private async _dispose() {\n const contextPromise = this._contextPromise;\n this._contextPromise = undefined;\n if (!contextPromise)\n return;\n const { close } = await contextPromise;\n await close(async () => { });\n }\n}\n\nasync function computeTracesDir(config: FullConfig, clientInfo: ClientInfo): Promise<string | undefined> {\n if (!config.saveTrace && !config.capabilities?.includes('tracing'))\n return;\n return await outputFile(config, clientInfo, `traces`, { origin: 'code', reason: 'Collecting trace' });\n}\n\nasync function computeHarDir(clientInfo: ClientInfo, runId?: string): Promise<string> {\n const finalRunId = runId || crypto.randomUUID();\n const harDir = path.join(os.homedir(), '.reverse-api-engineer', 'runs', 'testing-playwright', finalRunId);\n await fs.promises.mkdir(harDir, { recursive: true });\n const harPath = path.join(harDir, 'network.har');\n // eslint-disable-next-line no-console\n console.error(`HAR recording enabled. HAR file will be saved to: ${harPath}`);\n return harDir;\n}\n\nfunction browserContextOptionsFromConfig(config: FullConfig, harDir?: string): playwright.BrowserContextOptions {\n const result = { ...config.browser.contextOptions };\n if (config.saveVideo) {\n result.recordVideo = {\n dir: tmpDir(),\n size: config.saveVideo,\n };\n }\n if (harDir) {\n result.recordHar = {\n path: path.join(harDir, 'network.har'),\n content: 'attach',\n mode: 'full',\n };\n }\n return result;\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["import_server", "fs", "path", "net", "crypto", "os"]
7
+ }
@@ -0,0 +1,78 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+ var browserServerBackend_exports = {};
20
+ __export(browserServerBackend_exports, {
21
+ BrowserServerBackend: () => BrowserServerBackend
22
+ });
23
+ module.exports = __toCommonJS(browserServerBackend_exports);
24
+ var import_context = require("./context");
25
+ var import_log = require("../log");
26
+ var import_response = require("./response");
27
+ var import_sessionLog = require("./sessionLog");
28
+ var import_tools = require("./tools");
29
+ var import_tool = require("../sdk/tool");
30
+ class BrowserServerBackend {
31
+ constructor(config, factory) {
32
+ this._config = config;
33
+ this._browserContextFactory = factory;
34
+ this._tools = (0, import_tools.filteredTools)(config);
35
+ }
36
+ async initialize(clientInfo) {
37
+ this._sessionLog = this._config.saveSession ? await import_sessionLog.SessionLog.create(this._config, clientInfo) : void 0;
38
+ this._context = new import_context.Context({
39
+ config: this._config,
40
+ browserContextFactory: this._browserContextFactory,
41
+ sessionLog: this._sessionLog,
42
+ clientInfo
43
+ });
44
+ }
45
+ async listTools() {
46
+ return this._tools.map((tool) => (0, import_tool.toMcpTool)(tool.schema));
47
+ }
48
+ async callTool(name, rawArguments) {
49
+ const tool = this._tools.find((tool2) => tool2.schema.name === name);
50
+ if (!tool)
51
+ throw new Error(`Tool "${name}" not found`);
52
+ const parsedArguments = tool.schema.inputSchema.parse(rawArguments || {});
53
+ const context = this._context;
54
+ const response = new import_response.Response(context, name, parsedArguments);
55
+ response.logBegin();
56
+ context.setRunningTool(name);
57
+ try {
58
+ await tool.handle(context, parsedArguments, response);
59
+ await response.finish();
60
+ this._sessionLog?.logResponse(response);
61
+ } catch (error) {
62
+ response.addError(String(error));
63
+ } finally {
64
+ context.setRunningTool(void 0);
65
+ }
66
+ response.logEnd();
67
+ const _meta = rawArguments?._meta;
68
+ return response.serialize({ _meta });
69
+ }
70
+ serverClosed() {
71
+ void this._context?.dispose().catch(import_log.logUnhandledError);
72
+ }
73
+ }
74
+ // Annotate the CommonJS export names for ESM import in node:
75
+ 0 && (module.exports = {
76
+ BrowserServerBackend
77
+ });
78
+ //# sourceMappingURL=browserServerBackend.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/mcp/browser/browserServerBackend.ts"],
4
+ "sourcesContent": ["/**\n * Copyright (c) Microsoft Corporation.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { FullConfig } from './config';\nimport { Context } from './context';\nimport { logUnhandledError } from '../log';\nimport { Response } from './response';\nimport { SessionLog } from './sessionLog';\nimport { filteredTools } from './tools';\nimport { toMcpTool } from '../sdk/tool';\n\nimport type { Tool } from './tools/tool';\nimport type { BrowserContextFactory } from './browserContextFactory';\nimport type * as mcpServer from '../sdk/server';\nimport type { ServerBackend } from '../sdk/server';\n\nexport class BrowserServerBackend implements ServerBackend {\n private _tools: Tool[];\n private _context: Context | undefined;\n private _sessionLog: SessionLog | undefined;\n private _config: FullConfig;\n private _browserContextFactory: BrowserContextFactory;\n\n constructor(config: FullConfig, factory: BrowserContextFactory) {\n this._config = config;\n this._browserContextFactory = factory;\n this._tools = filteredTools(config);\n }\n\n async initialize(clientInfo: mcpServer.ClientInfo): Promise<void> {\n this._sessionLog = this._config.saveSession ? await SessionLog.create(this._config, clientInfo) : undefined;\n this._context = new Context({\n config: this._config,\n browserContextFactory: this._browserContextFactory,\n sessionLog: this._sessionLog,\n clientInfo,\n });\n }\n\n async listTools(): Promise<mcpServer.Tool[]> {\n return this._tools.map(tool => toMcpTool(tool.schema));\n }\n\n async callTool(name: string, rawArguments: mcpServer.CallToolRequest['params']['arguments']) {\n const tool = this._tools.find(tool => tool.schema.name === name)!;\n if (!tool)\n throw new Error(`Tool \"${name}\" not found`);\n const parsedArguments = tool.schema.inputSchema.parse(rawArguments || {});\n const context = this._context!;\n const response = new Response(context, name, parsedArguments);\n response.logBegin();\n context.setRunningTool(name);\n try {\n await tool.handle(context, parsedArguments, response);\n await response.finish();\n this._sessionLog?.logResponse(response);\n } catch (error: any) {\n response.addError(String(error));\n } finally {\n context.setRunningTool(undefined);\n }\n response.logEnd();\n const _meta = rawArguments?._meta as object | undefined;\n return response.serialize({ _meta });\n }\n\n serverClosed() {\n void this._context?.dispose().catch(logUnhandledError);\n }\n}\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAiBA,qBAAwB;AACxB,iBAAkC;AAClC,sBAAyB;AACzB,wBAA2B;AAC3B,mBAA8B;AAC9B,kBAA0B;AAOnB,MAAM,qBAA8C;AAAA,EAOzD,YAAY,QAAoB,SAAgC;AAC9D,SAAK,UAAU;AACf,SAAK,yBAAyB;AAC9B,SAAK,aAAS,4BAAc,MAAM;AAAA,EACpC;AAAA,EAEA,MAAM,WAAW,YAAiD;AAChE,SAAK,cAAc,KAAK,QAAQ,cAAc,MAAM,6BAAW,OAAO,KAAK,SAAS,UAAU,IAAI;AAClG,SAAK,WAAW,IAAI,uBAAQ;AAAA,MAC1B,QAAQ,KAAK;AAAA,MACb,uBAAuB,KAAK;AAAA,MAC5B,YAAY,KAAK;AAAA,MACjB;AAAA,IACF,CAAC;AAAA,EACH;AAAA,EAEA,MAAM,YAAuC;AAC3C,WAAO,KAAK,OAAO,IAAI,cAAQ,uBAAU,KAAK,MAAM,CAAC;AAAA,EACvD;AAAA,EAEA,MAAM,SAAS,MAAc,cAAgE;AAC3F,UAAM,OAAO,KAAK,OAAO,KAAK,CAAAA,UAAQA,MAAK,OAAO,SAAS,IAAI;AAC/D,QAAI,CAAC;AACH,YAAM,IAAI,MAAM,SAAS,IAAI,aAAa;AAC5C,UAAM,kBAAkB,KAAK,OAAO,YAAY,MAAM,gBAAgB,CAAC,CAAC;AACxE,UAAM,UAAU,KAAK;AACrB,UAAM,WAAW,IAAI,yBAAS,SAAS,MAAM,eAAe;AAC5D,aAAS,SAAS;AAClB,YAAQ,eAAe,IAAI;AAC3B,QAAI;AACF,YAAM,KAAK,OAAO,SAAS,iBAAiB,QAAQ;AACpD,YAAM,SAAS,OAAO;AACtB,WAAK,aAAa,YAAY,QAAQ;AAAA,IACxC,SAAS,OAAY;AACnB,eAAS,SAAS,OAAO,KAAK,CAAC;AAAA,IACjC,UAAE;AACA,cAAQ,eAAe,MAAS;AAAA,IAClC;AACA,aAAS,OAAO;AAChB,UAAM,QAAQ,cAAc;AAC5B,WAAO,SAAS,UAAU,EAAE,MAAM,CAAC;AAAA,EACrC;AAAA,EAEA,eAAe;AACb,SAAK,KAAK,UAAU,QAAQ,EAAE,MAAM,4BAAiB;AAAA,EACvD;AACF;",
6
+ "names": ["tool"]
7
+ }