@melony/plugin-browser 0.1.1 → 0.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.js +144 -0
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -129,6 +129,145 @@ var browserPlugin = (options = {}) => {
|
|
|
129
129
|
return page;
|
|
130
130
|
}
|
|
131
131
|
return (builder) => {
|
|
132
|
+
async function* yieldStateUpdate(page2) {
|
|
133
|
+
try {
|
|
134
|
+
const url = page2.url();
|
|
135
|
+
const title = await page2.title();
|
|
136
|
+
const screenshot = await page2.screenshot({ type: "jpeg", quality: 60 });
|
|
137
|
+
const base64 = screenshot.toString("base64");
|
|
138
|
+
const pages = context ? await context.pages() : [page2];
|
|
139
|
+
const tabs = await Promise.all(pages.map(async (p, i) => ({
|
|
140
|
+
index: i,
|
|
141
|
+
title: await p.title().catch(() => "Untitled"),
|
|
142
|
+
url: p.url(),
|
|
143
|
+
isActive: p === page2
|
|
144
|
+
})));
|
|
145
|
+
yield {
|
|
146
|
+
type: "browser:state-update",
|
|
147
|
+
data: {
|
|
148
|
+
url,
|
|
149
|
+
title,
|
|
150
|
+
screenshot: base64,
|
|
151
|
+
tabCount: pages.length,
|
|
152
|
+
tabs
|
|
153
|
+
}
|
|
154
|
+
};
|
|
155
|
+
yield {
|
|
156
|
+
type: "ui",
|
|
157
|
+
data: {
|
|
158
|
+
type: "card",
|
|
159
|
+
props: {
|
|
160
|
+
title: title || "Browser View",
|
|
161
|
+
subtitle: url,
|
|
162
|
+
padding: "none"
|
|
163
|
+
},
|
|
164
|
+
children: [
|
|
165
|
+
{
|
|
166
|
+
type: "image",
|
|
167
|
+
props: {
|
|
168
|
+
src: `data:image/jpeg;base64,${base64}`,
|
|
169
|
+
alt: "Browser Screenshot",
|
|
170
|
+
width: "full"
|
|
171
|
+
}
|
|
172
|
+
},
|
|
173
|
+
{
|
|
174
|
+
type: "row",
|
|
175
|
+
props: { justify: "between", padding: "sm" },
|
|
176
|
+
children: [
|
|
177
|
+
{
|
|
178
|
+
type: "text",
|
|
179
|
+
props: { value: `Tabs open: ${pages.length}`, size: "xs", color: "muted" }
|
|
180
|
+
},
|
|
181
|
+
{
|
|
182
|
+
type: "button",
|
|
183
|
+
props: {
|
|
184
|
+
label: "Cleanup Tabs",
|
|
185
|
+
size: "xs",
|
|
186
|
+
variant: "outline",
|
|
187
|
+
onClickAction: { type: "browser:cleanup" }
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
]
|
|
191
|
+
}
|
|
192
|
+
]
|
|
193
|
+
}
|
|
194
|
+
};
|
|
195
|
+
} catch (e) {
|
|
196
|
+
console.error("Failed to yield state update", e);
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
builder.on("browser:poll_state", async function* (event) {
|
|
200
|
+
const p = await ensurePage();
|
|
201
|
+
yield* yieldStateUpdate(p);
|
|
202
|
+
});
|
|
203
|
+
builder.on("browser:cleanup", async function* (event) {
|
|
204
|
+
if (context) {
|
|
205
|
+
const pages = await context.pages();
|
|
206
|
+
for (const p2 of pages) {
|
|
207
|
+
if (p2 !== page && !p2.isClosed()) {
|
|
208
|
+
await p2.close().catch(() => {
|
|
209
|
+
});
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
const p = await ensurePage();
|
|
214
|
+
yield* yieldStateUpdate(p);
|
|
215
|
+
});
|
|
216
|
+
builder.on("action:browser_listTabs", async function* (event) {
|
|
217
|
+
const { toolCallId } = event.data;
|
|
218
|
+
try {
|
|
219
|
+
await ensurePage();
|
|
220
|
+
const pages = context ? await context.pages() : [page];
|
|
221
|
+
const tabs = await Promise.all(pages.map(async (p, i) => ({
|
|
222
|
+
index: i,
|
|
223
|
+
title: await p.title().catch(() => "Untitled"),
|
|
224
|
+
url: p.url(),
|
|
225
|
+
isActive: p === page
|
|
226
|
+
})));
|
|
227
|
+
yield {
|
|
228
|
+
type: "action:result",
|
|
229
|
+
data: {
|
|
230
|
+
action: "browser_listTabs",
|
|
231
|
+
toolCallId,
|
|
232
|
+
result: { success: true, tabs }
|
|
233
|
+
}
|
|
234
|
+
};
|
|
235
|
+
} catch (error) {
|
|
236
|
+
yield {
|
|
237
|
+
type: "action:result",
|
|
238
|
+
data: { action: "browser_listTabs", toolCallId, result: { error: error.message } }
|
|
239
|
+
};
|
|
240
|
+
}
|
|
241
|
+
});
|
|
242
|
+
builder.on("action:browser_closeTab", async function* (event) {
|
|
243
|
+
const { index, toolCallId } = event.data;
|
|
244
|
+
try {
|
|
245
|
+
await ensurePage();
|
|
246
|
+
const pages = context ? await context.pages() : [page];
|
|
247
|
+
if (index >= 0 && index < pages.length) {
|
|
248
|
+
const tabToClose = pages[index];
|
|
249
|
+
if (tabToClose === page) {
|
|
250
|
+
page = null;
|
|
251
|
+
}
|
|
252
|
+
await tabToClose.close();
|
|
253
|
+
}
|
|
254
|
+
const p = await ensurePage();
|
|
255
|
+
yield* yieldStateUpdate(p);
|
|
256
|
+
yield {
|
|
257
|
+
type: "action:result",
|
|
258
|
+
data: {
|
|
259
|
+
action: "browser_closeTab",
|
|
260
|
+
toolCallId,
|
|
261
|
+
result: { success: true }
|
|
262
|
+
}
|
|
263
|
+
};
|
|
264
|
+
} catch (error) {
|
|
265
|
+
yield {
|
|
266
|
+
type: "action:result",
|
|
267
|
+
data: { action: "browser_closeTab", toolCallId, result: { error: error.message } }
|
|
268
|
+
};
|
|
269
|
+
}
|
|
270
|
+
});
|
|
132
271
|
builder.on("action:browser_navigate", async function* (event) {
|
|
133
272
|
const { url, waitUntil, toolCallId } = event.data;
|
|
134
273
|
yield {
|
|
@@ -143,6 +282,7 @@ var browserPlugin = (options = {}) => {
|
|
|
143
282
|
try {
|
|
144
283
|
const page2 = await ensurePage();
|
|
145
284
|
await page2.goto(url, { waitUntil });
|
|
285
|
+
yield* yieldStateUpdate(page2);
|
|
146
286
|
yield {
|
|
147
287
|
type: "ui",
|
|
148
288
|
data: {
|
|
@@ -191,6 +331,7 @@ var browserPlugin = (options = {}) => {
|
|
|
191
331
|
const page2 = await ensurePage();
|
|
192
332
|
const buffer = await page2.screenshot({ fullPage, type: "png" });
|
|
193
333
|
const base64 = buffer.toString("base64");
|
|
334
|
+
yield* yieldStateUpdate(page2);
|
|
194
335
|
yield {
|
|
195
336
|
type: "action:result",
|
|
196
337
|
data: {
|
|
@@ -240,6 +381,7 @@ var browserPlugin = (options = {}) => {
|
|
|
240
381
|
try {
|
|
241
382
|
const page2 = await ensurePage();
|
|
242
383
|
await page2.click(selector);
|
|
384
|
+
yield* yieldStateUpdate(page2);
|
|
243
385
|
yield {
|
|
244
386
|
type: "action:result",
|
|
245
387
|
data: { action: "browser_click", toolCallId, result: { success: true } }
|
|
@@ -274,6 +416,7 @@ var browserPlugin = (options = {}) => {
|
|
|
274
416
|
try {
|
|
275
417
|
const page2 = await ensurePage();
|
|
276
418
|
await page2.type(selector, text, { delay });
|
|
419
|
+
yield* yieldStateUpdate(page2);
|
|
277
420
|
yield {
|
|
278
421
|
type: "action:result",
|
|
279
422
|
data: { action: "browser_type", toolCallId, result: { success: true } }
|
|
@@ -365,6 +508,7 @@ var browserPlugin = (options = {}) => {
|
|
|
365
508
|
try {
|
|
366
509
|
const page2 = await ensurePage();
|
|
367
510
|
const result = await page2.evaluate(script);
|
|
511
|
+
yield* yieldStateUpdate(page2);
|
|
368
512
|
yield {
|
|
369
513
|
type: "action:result",
|
|
370
514
|
data: { action: "browser_evaluate", toolCallId, result: { success: true, result } }
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["import { MelonyPlugin } from \"melony\";\nimport { z } from \"zod\";\nimport { chromium, Browser, BrowserContext, Page } from \"playwright\";\n\n// Shared state to avoid \"ProcessSingleton\" errors in long-running processes (like Express)\n// mapping userDataDir -> Context\nconst persistentContexts = new Map<string, BrowserContext>();\n// mapping channel/options -> Browser\nconst sharedBrowsers = new Map<string, Browser>();\n\nexport const browserToolDefinitions = {\n browser_navigate: {\n description: \"Navigate to a URL and wait for the page to load\",\n inputSchema: z.object({\n url: z.string().describe(\"The URL to navigate to\"),\n waitUntil: z.enum([\"load\", \"domcontentloaded\", \"networkidle\", \"commit\"]).default(\"load\").describe(\"When to consider navigation finished\"),\n }),\n },\n browser_screenshot: {\n description: \"Take a screenshot of the current page and return it as a base64 image\",\n inputSchema: z.object({\n fullPage: z.boolean().default(false).describe(\"Whether to take a screenshot of the full scrollable page\"),\n }),\n },\n browser_click: {\n description: \"Click an element on the page\",\n inputSchema: z.object({\n selector: z.string().describe(\"CSS selector or XPath of the element to click\"),\n }),\n },\n browser_type: {\n description: \"Type text into an input field\",\n inputSchema: z.object({\n selector: z.string().describe(\"CSS selector or XPath of the element to type into\"),\n text: z.string().describe(\"The text to type\"),\n delay: z.number().optional().describe(\"Delay between keystrokes in milliseconds\"),\n }),\n },\n browser_getContent: {\n description: \"Get the content of the current page\",\n inputSchema: z.object({\n format: z.enum([\"text\", \"html\", \"markdown\"]).default(\"text\").describe(\"The format to return the content in\"),\n }),\n },\n browser_evaluate: {\n description: \"Execute JavaScript code in the context of the page\",\n inputSchema: z.object({\n script: z.string().describe(\"The JavaScript code to execute\"),\n }),\n },\n};\n\nexport interface BrowserPluginOptions {\n headless?: boolean;\n /**\n * Maximum number of characters to keep when getting page content.\n * If exceeded, the content will be truncated from the middle.\n * Default: 10000\n */\n maxContentLength?: number;\n /**\n * Path to user data directory for persistent browser sessions.\n * When provided, the browser will use launchPersistentContext which\n * preserves cookies, localStorage, and login sessions across runs.\n * \n * First run with headless: false to manually log in, then subsequent\n * runs will automatically have your session.\n */\n userDataDir?: string;\n /**\n * Browser channel to use (e.g. 'chrome', 'chrome-beta', 'msedge').\n * Using 'chrome' can help bypass \"browser not safe\" errors on Google/YouTube.\n */\n channel?: string;\n}\n\n/**\n * Truncates a string by keeping the first and last N characters.\n */\nfunction truncate(str: string | undefined | null, maxChars: number): string | undefined | null {\n if (!str || str.length <= maxChars) return str;\n const half = Math.floor(maxChars / 2);\n const truncatedCount = str.length - maxChars;\n return `${str.slice(0, half)}\\n\\n[... ${truncatedCount} characters truncated ...]\\n\\n${str.slice(-half)}`;\n}\n\nexport const browserPlugin = (options: BrowserPluginOptions = {}): MelonyPlugin<any, any> => {\n let browser: Browser | null = null;\n let context: BrowserContext | null = null;\n let page: Page | null = null;\n\n const { headless = true, maxContentLength = 10000, userDataDir, channel } = options;\n\n async function ensurePage(): Promise<Page> {\n // If page exists but is closed, clear it\n if (page && page.isClosed()) {\n page = null;\n }\n\n if (userDataDir) {\n // Use persistent context for session persistence (cookies, localStorage, etc.)\n const isContextUsable = async (ctx: BrowserContext | null) => {\n if (!ctx) return false;\n try {\n await ctx.pages();\n return true;\n } catch {\n return false;\n }\n };\n\n if (!context || !(await isContextUsable(context))) {\n // Check if we already have a context for this directory in this process\n if (persistentContexts.has(userDataDir)) {\n context = persistentContexts.get(userDataDir)!;\n }\n\n // If context is still null or not usable, launch it\n if (!context || !(await isContextUsable(context))) {\n context = await chromium.launchPersistentContext(userDataDir, {\n headless,\n channel,\n viewport: { width: 1280, height: 720 },\n ignoreDefaultArgs: [\"--enable-automation\"],\n args: [\n \"--disable-blink-features=AutomationControlled\",\n \"--no-sandbox\",\n \"--disable-infobars\",\n \"--window-position=0,0\",\n \"--ignore-certifcate-errors\",\n \"--ignore-certifcate-errors-spki-list\",\n \"--user-agent=Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36\",\n ],\n });\n context.on(\"close\", () => persistentContexts.delete(userDataDir));\n persistentContexts.set(userDataDir, context);\n }\n \n // Ensure we have a page\n if (!page || page.isClosed()) {\n page = context.pages()[0] || await context.newPage();\n }\n }\n } else {\n // Standard non-persistent browser\n const isBrowserUsable = (b: Browser | null) => b && b.isConnected();\n \n if (!browser || !isBrowserUsable(browser)) {\n const browserKey = `${channel || \"default\"}-${headless}`;\n if (sharedBrowsers.has(browserKey)) {\n browser = sharedBrowsers.get(browserKey)!;\n }\n \n if (!browser || !isBrowserUsable(browser)) {\n browser = await chromium.launch({\n headless,\n channel,\n ignoreDefaultArgs: [\"--enable-automation\"],\n args: [\"--disable-blink-features=AutomationControlled\"],\n });\n browser.on(\"disconnected\", () => sharedBrowsers.delete(browserKey));\n sharedBrowsers.set(browserKey, browser);\n }\n }\n\n if (!page || page.isClosed()) {\n context = await browser.newContext({\n viewport: { width: 1280, height: 720 },\n userAgent: \"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36\",\n });\n page = await context.newPage();\n }\n }\n return page!;\n }\n\n return (builder) => {\n builder.on(\"action:browser_navigate\", async function* (event) {\n const { url, waitUntil, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Navigating to: ${url}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n await page.goto(url, { waitUntil });\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Navigated to: ${page.url()}`,\n }\n },\n };\n\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_navigate\",\n toolCallId,\n result: { success: true, url: page.url(), title: await page.title() },\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_navigate\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Navigation failed: ${error.message}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_screenshot\", async function* (event) {\n const { fullPage, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Taking screenshot`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n const buffer = await page.screenshot({ fullPage, type: \"png\" });\n const base64 = buffer.toString(\"base64\");\n\n // Return both the raw result and a UI event to show the screenshot\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_screenshot\",\n toolCallId,\n result: { success: true, format: \"png\", message: \"Screenshot taken successfully\" },\n },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: \"image\",\n props: {\n src: `data:image/png;base64,${base64}`,\n alt: \"Browser Screenshot\",\n width: \"full\",\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_screenshot\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Screenshot failed: ${error.message}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_click\", async function* (event) {\n const { selector, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Clicking: ${selector}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n await page.click(selector);\n yield {\n type: \"action:result\",\n data: { action: \"browser_click\", toolCallId, result: { success: true } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Clicked: ${selector}`,\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_click\", toolCallId, result: { error: error.message } },\n };\n }\n });\n\n builder.on(\"action:browser_type\", async function* (event) {\n const { selector, text, delay, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Typing: ${selector}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n await page.type(selector, text, { delay });\n yield {\n type: \"action:result\",\n data: { action: \"browser_type\", toolCallId, result: { success: true } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Typed: ${selector}`,\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_type\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Typed: ${selector}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_getContent\", async function* (event) {\n const { format, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Getting content: ${format}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n let content = \"\";\n if (format === \"html\") {\n content = await page.content();\n } else if (format === \"text\") {\n content = await page.evaluate(() => document.body.innerText);\n } else if (format === \"markdown\") {\n // Basic markdown conversion if needed, for now just text\n content = await page.evaluate(() => document.body.innerText);\n }\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_getContent\",\n toolCallId,\n result: {\n success: true,\n content: truncate(content, maxContentLength)\n }\n },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Content: ${content}`,\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_getContent\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Content failed: ${error.message}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_evaluate\", async function* (event) {\n const { script, toolCallId } = event.data;\n try {\n const page = await ensurePage();\n const result = await page.evaluate(script);\n yield {\n type: \"action:result\",\n data: { action: \"browser_evaluate\", toolCallId, result: { success: true, result } },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_evaluate\", toolCallId, result: { error: error.message } },\n };\n }\n });\n };\n};\n"],"mappings":"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","names":["page"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["import { MelonyPlugin } from \"melony\";\nimport { z } from \"zod\";\nimport { chromium, Browser, BrowserContext, Page } from \"playwright\";\n\n// Shared state to avoid \"ProcessSingleton\" errors in long-running processes (like Express)\n// mapping userDataDir -> Context\nconst persistentContexts = new Map<string, BrowserContext>();\n// mapping channel/options -> Browser\nconst sharedBrowsers = new Map<string, Browser>();\n\nexport const browserToolDefinitions = {\n browser_navigate: {\n description: \"Navigate to a URL and wait for the page to load\",\n inputSchema: z.object({\n url: z.string().describe(\"The URL to navigate to\"),\n waitUntil: z.enum([\"load\", \"domcontentloaded\", \"networkidle\", \"commit\"]).default(\"load\").describe(\"When to consider navigation finished\"),\n }),\n },\n browser_screenshot: {\n description: \"Take a screenshot of the current page and return it as a base64 image\",\n inputSchema: z.object({\n fullPage: z.boolean().default(false).describe(\"Whether to take a screenshot of the full scrollable page\"),\n }),\n },\n browser_click: {\n description: \"Click an element on the page\",\n inputSchema: z.object({\n selector: z.string().describe(\"CSS selector or XPath of the element to click\"),\n }),\n },\n browser_type: {\n description: \"Type text into an input field\",\n inputSchema: z.object({\n selector: z.string().describe(\"CSS selector or XPath of the element to type into\"),\n text: z.string().describe(\"The text to type\"),\n delay: z.number().optional().describe(\"Delay between keystrokes in milliseconds\"),\n }),\n },\n browser_getContent: {\n description: \"Get the content of the current page\",\n inputSchema: z.object({\n format: z.enum([\"text\", \"html\", \"markdown\"]).default(\"text\").describe(\"The format to return the content in\"),\n }),\n },\n browser_evaluate: {\n description: \"Execute JavaScript code in the context of the page\",\n inputSchema: z.object({\n script: z.string().describe(\"The JavaScript code to execute\"),\n }),\n }\n};\n\nexport interface BrowserPluginOptions {\n headless?: boolean;\n /**\n * Maximum number of characters to keep when getting page content.\n * If exceeded, the content will be truncated from the middle.\n * Default: 10000\n */\n maxContentLength?: number;\n /**\n * Path to user data directory for persistent browser sessions.\n * When provided, the browser will use launchPersistentContext which\n * preserves cookies, localStorage, and login sessions across runs.\n * \n * First run with headless: false to manually log in, then subsequent\n * runs will automatically have your session.\n */\n userDataDir?: string;\n /**\n * Browser channel to use (e.g. 'chrome', 'chrome-beta', 'msedge').\n * Using 'chrome' can help bypass \"browser not safe\" errors on Google/YouTube.\n */\n channel?: string;\n}\n\n/**\n * Truncates a string by keeping the first and last N characters.\n */\nfunction truncate(str: string | undefined | null, maxChars: number): string | undefined | null {\n if (!str || str.length <= maxChars) return str;\n const half = Math.floor(maxChars / 2);\n const truncatedCount = str.length - maxChars;\n return `${str.slice(0, half)}\\n\\n[... ${truncatedCount} characters truncated ...]\\n\\n${str.slice(-half)}`;\n}\n\nexport const browserPlugin = (options: BrowserPluginOptions = {}): MelonyPlugin<any, any> => {\n let browser: Browser | null = null;\n let context: BrowserContext | null = null;\n let page: Page | null = null;\n\n const { headless = true, maxContentLength = 10000, userDataDir, channel } = options;\n\n async function ensurePage(): Promise<Page> {\n // If page exists but is closed, clear it\n if (page && page.isClosed()) {\n page = null;\n }\n\n if (userDataDir) {\n // Use persistent context for session persistence (cookies, localStorage, etc.)\n const isContextUsable = async (ctx: BrowserContext | null) => {\n if (!ctx) return false;\n try {\n await ctx.pages();\n return true;\n } catch {\n return false;\n }\n };\n\n if (!context || !(await isContextUsable(context))) {\n // Check if we already have a context for this directory in this process\n if (persistentContexts.has(userDataDir)) {\n context = persistentContexts.get(userDataDir)!;\n }\n\n // If context is still null or not usable, launch it\n if (!context || !(await isContextUsable(context))) {\n context = await chromium.launchPersistentContext(userDataDir, {\n headless,\n channel,\n viewport: { width: 1280, height: 720 },\n ignoreDefaultArgs: [\"--enable-automation\"],\n args: [\n \"--disable-blink-features=AutomationControlled\",\n \"--no-sandbox\",\n \"--disable-infobars\",\n \"--window-position=0,0\",\n \"--ignore-certifcate-errors\",\n \"--ignore-certifcate-errors-spki-list\",\n \"--user-agent=Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36\",\n ],\n });\n context.on(\"close\", () => persistentContexts.delete(userDataDir));\n persistentContexts.set(userDataDir, context);\n }\n\n // Ensure we have a page\n if (!page || page.isClosed()) {\n page = context.pages()[0] || await context.newPage();\n }\n }\n } else {\n // Standard non-persistent browser\n const isBrowserUsable = (b: Browser | null) => b && b.isConnected();\n\n if (!browser || !isBrowserUsable(browser)) {\n const browserKey = `${channel || \"default\"}-${headless}`;\n if (sharedBrowsers.has(browserKey)) {\n browser = sharedBrowsers.get(browserKey)!;\n }\n\n if (!browser || !isBrowserUsable(browser)) {\n browser = await chromium.launch({\n headless,\n channel,\n ignoreDefaultArgs: [\"--enable-automation\"],\n args: [\"--disable-blink-features=AutomationControlled\"],\n });\n browser.on(\"disconnected\", () => sharedBrowsers.delete(browserKey));\n sharedBrowsers.set(browserKey, browser);\n }\n }\n\n if (!page || page.isClosed()) {\n context = await browser.newContext({\n viewport: { width: 1280, height: 720 },\n userAgent: \"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36\",\n });\n page = await context.newPage();\n }\n }\n return page!;\n }\n\n return (builder) => {\n async function* yieldStateUpdate(page: Page) {\n try {\n const url = page.url();\n const title = await page.title();\n const screenshot = await page.screenshot({ type: \"jpeg\", quality: 60 });\n const base64 = screenshot.toString(\"base64\");\n \n // Get all pages to show tab count in UI\n const pages = context ? await context.pages() : [page];\n const tabs = await Promise.all(pages.map(async (p, i) => ({\n index: i,\n title: await p.title().catch(() => \"Untitled\"),\n url: p.url(),\n isActive: p === page\n })));\n\n // Yield for potential sidebars/background state\n yield {\n type: \"browser:state-update\",\n data: {\n url,\n title,\n screenshot: base64,\n tabCount: pages.length,\n tabs,\n },\n };\n\n // Yield as a UI element for the chat bubble\n yield {\n type: \"ui\",\n data: {\n type: \"card\",\n props: {\n title: title || \"Browser View\",\n subtitle: url,\n padding: \"none\",\n },\n children: [\n {\n type: \"image\",\n props: {\n src: `data:image/jpeg;base64,${base64}`,\n alt: \"Browser Screenshot\",\n width: \"full\",\n },\n },\n {\n type: \"row\",\n props: { justify: \"between\", padding: \"sm\" },\n children: [\n {\n type: \"text\",\n props: { value: `Tabs open: ${pages.length}`, size: \"xs\", color: \"muted\" }\n },\n {\n type: \"button\",\n props: { \n label: \"Cleanup Tabs\", \n size: \"xs\", \n variant: \"outline\",\n onClickAction: { type: \"browser:cleanup\" }\n }\n }\n ]\n }\n ],\n },\n };\n } catch (e) {\n console.error(\"Failed to yield state update\", e);\n }\n }\n\n builder.on(\"browser:poll_state\", async function* (event) {\n const p = await ensurePage();\n yield* yieldStateUpdate(p);\n });\n\n builder.on(\"browser:cleanup\", async function* (event) {\n if (context) {\n const pages = await context.pages();\n // Keep the current page, close all others\n for (const p of pages) {\n if (p !== page && !p.isClosed()) {\n await p.close().catch(() => {});\n }\n }\n }\n const p = await ensurePage();\n yield* yieldStateUpdate(p);\n });\n\n builder.on(\"action:browser_listTabs\", async function* (event) {\n const { toolCallId } = event.data;\n try {\n await ensurePage();\n const pages = context ? await context.pages() : [page!];\n const tabs = await Promise.all(pages.map(async (p, i) => ({\n index: i,\n title: await p.title().catch(() => \"Untitled\"),\n url: p.url(),\n isActive: p === page\n })));\n\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_listTabs\",\n toolCallId,\n result: { success: true, tabs },\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_listTabs\", toolCallId, result: { error: error.message } },\n };\n }\n });\n\n builder.on(\"action:browser_closeTab\", async function* (event) {\n const { index, toolCallId } = event.data;\n try {\n await ensurePage();\n const pages = context ? await context.pages() : [page!];\n \n if (index >= 0 && index < pages.length) {\n const tabToClose = pages[index];\n if (tabToClose === page) {\n // If we're closing the active page, we need to pick a new one\n page = null;\n }\n await tabToClose.close();\n }\n\n const p = await ensurePage();\n yield* yieldStateUpdate(p);\n\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_closeTab\",\n toolCallId,\n result: { success: true },\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_closeTab\", toolCallId, result: { error: error.message } },\n };\n }\n });\n\n builder.on(\"action:browser_navigate\", async function* (event) {\n const { url, waitUntil, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Navigating to: ${url}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n await page.goto(url, { waitUntil });\n\n yield* yieldStateUpdate(page);\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Navigated to: ${page.url()}`,\n }\n },\n };\n\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_navigate\",\n toolCallId,\n result: { success: true, url: page.url(), title: await page.title() },\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_navigate\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Navigation failed: ${error.message}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_screenshot\", async function* (event) {\n const { fullPage, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Taking screenshot`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n const buffer = await page.screenshot({ fullPage, type: \"png\" });\n const base64 = buffer.toString(\"base64\");\n\n yield* yieldStateUpdate(page);\n\n // Return both the raw result and a UI event to show the screenshot\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_screenshot\",\n toolCallId,\n result: { success: true, format: \"png\", message: \"Screenshot taken successfully\" },\n },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: \"image\",\n props: {\n src: `data:image/png;base64,${base64}`,\n alt: \"Browser Screenshot\",\n width: \"full\",\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_screenshot\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Screenshot failed: ${error.message}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_click\", async function* (event) {\n const { selector, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Clicking: ${selector}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n await page.click(selector);\n\n yield* yieldStateUpdate(page);\n\n yield {\n type: \"action:result\",\n data: { action: \"browser_click\", toolCallId, result: { success: true } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Clicked: ${selector}`,\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_click\", toolCallId, result: { error: error.message } },\n };\n }\n });\n\n builder.on(\"action:browser_type\", async function* (event) {\n const { selector, text, delay, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Typing: ${selector}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n await page.type(selector, text, { delay });\n\n yield* yieldStateUpdate(page);\n\n yield {\n type: \"action:result\",\n data: { action: \"browser_type\", toolCallId, result: { success: true } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Typed: ${selector}`,\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_type\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Typed: ${selector}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_getContent\", async function* (event) {\n const { format, toolCallId } = event.data;\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Getting content: ${format}`,\n }\n },\n };\n\n try {\n const page = await ensurePage();\n let content = \"\";\n if (format === \"html\") {\n content = await page.content();\n } else if (format === \"text\") {\n content = await page.evaluate(() => document.body.innerText);\n } else if (format === \"markdown\") {\n // Basic markdown conversion if needed, for now just text\n content = await page.evaluate(() => document.body.innerText);\n }\n yield {\n type: \"action:result\",\n data: {\n action: \"browser_getContent\",\n toolCallId,\n result: {\n success: true,\n content: truncate(content, maxContentLength)\n }\n },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Content: ${content}`,\n }\n },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_getContent\", toolCallId, result: { error: error.message } },\n };\n\n yield {\n type: \"ui\",\n data: {\n type: 'text',\n props: {\n value: `Content failed: ${error.message}`,\n }\n },\n };\n }\n });\n\n builder.on(\"action:browser_evaluate\", async function* (event) {\n const { script, toolCallId } = event.data;\n try {\n const page = await ensurePage();\n const result = await page.evaluate(script);\n\n yield* yieldStateUpdate(page);\n\n yield {\n type: \"action:result\",\n data: { action: \"browser_evaluate\", toolCallId, result: { success: true, result } },\n };\n } catch (error: any) {\n yield {\n type: \"action:result\",\n data: { action: \"browser_evaluate\", toolCallId, result: { error: error.message } },\n };\n }\n });\n };\n};\n"],"mappings":"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","names":["page","p"]}
|