@mastra/browser-firecrawl 0.0.0 → 0.1.0-alpha.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/CHANGELOG.md ADDED
@@ -0,0 +1,34 @@
1
+ # @mastra/browser-firecrawl
2
+
3
+ ## 0.1.0-alpha.0
4
+
5
+ ### Minor Changes
6
+
7
+ - Initial release: Firecrawl Browser Sandbox integration for Mastra. ([#15724](https://github.com/mastra-ai/mastra/pull/15724))
8
+
9
+ `FirecrawlBrowser` extends `AgentBrowser` to run the same deterministic browser tools (snapshot+refs, 16 tools, Playwright over CDP) against Firecrawl's cloud-hosted Chrome instances instead of local or self-hosted browsers.
10
+
11
+ **Features:**
12
+ - Cloud-hosted Chrome via Firecrawl Browser Sandbox API
13
+ - Same tool surface as `@mastra/agent-browser` (~16 browser automation tools)
14
+ - Thread-scoped browser isolation (`scope: 'thread'`)
15
+ - Automatic session cleanup on close
16
+
17
+ **Usage:**
18
+
19
+ ```typescript
20
+ import { FirecrawlBrowser } from '@mastra/browser-firecrawl';
21
+
22
+ const browser = new FirecrawlBrowser({
23
+ firecrawlApiKey: process.env.FIRECRAWL_API_KEY,
24
+ scope: 'thread',
25
+ });
26
+
27
+ const agent = mastra.getAgent('my-agent', { browser });
28
+ ```
29
+
30
+ ### Patch Changes
31
+
32
+ - Updated dependencies [[`8ace89d`](https://github.com/mastra-ai/mastra/commit/8ace89df77f762e622d3b9f7f65ad7524350d050), [`fa63872`](https://github.com/mastra-ai/mastra/commit/fa6387280954e6b667bec5714b55ba082bc627ff), [`f07b646`](https://github.com/mastra-ai/mastra/commit/f07b64604ab7d25391179790b7fd4823df9e2dff), [`d8838ae`](https://github.com/mastra-ai/mastra/commit/d8838ae80b69780361693d27098f7f6684af12fe), [`40f9297`](https://github.com/mastra-ai/mastra/commit/40f9297003b921c62373d3e8d3a4bda76c9f6de3), [`0f0d1ba`](https://github.com/mastra-ai/mastra/commit/0f0d1ba67bfcb2204e571401662f1eceefc03357), [`8c31bcd`](https://github.com/mastra-ai/mastra/commit/8c31bcdb00e597880d5939b1b7d7566fbe5dacae), [`95b14cd`](https://github.com/mastra-ai/mastra/commit/95b14cdd820e86d97ac05fe568424c513a252e31), [`aa36be2`](https://github.com/mastra-ai/mastra/commit/aa36be23aa513b7dc53cb8ca16b7fab8f20e43ad), [`212c635`](https://github.com/mastra-ai/mastra/commit/212c635203e61d036ab41db8ff86c3893dc795b3), [`d8838ae`](https://github.com/mastra-ai/mastra/commit/d8838ae80b69780361693d27098f7f6684af12fe), [`9aa5a73`](https://github.com/mastra-ai/mastra/commit/9aa5a73e7e110f6e9365eec69364a33d5f03bb56), [`f73c789`](https://github.com/mastra-ai/mastra/commit/f73c789e8ef21561580395d2c410119cab5848c8), [`8bd16da`](https://github.com/mastra-ai/mastra/commit/8bd16da73a4cb874d739373643dbd6a6e7f88684), [`c8630f8`](https://github.com/mastra-ai/mastra/commit/c8630f80d4f40cb5d22e60ab162b618b1907167a), [`47f71dc`](https://github.com/mastra-ai/mastra/commit/47f71dc6fbcbd12d71e21a979e676e20a02bd77d), [`e191065`](https://github.com/mastra-ai/mastra/commit/e191065af6039cf6388e05aa2b84f6f5d69af4c9), [`50ceae2`](https://github.com/mastra-ai/mastra/commit/50ceae270878e2f8fb2b2c6c2faab09df0007c8a), [`8cdde58`](https://github.com/mastra-ai/mastra/commit/8cdde5875bbba6702d9df226f2b20232b8d75d6c), [`847ff1e`](https://github.com/mastra-ai/mastra/commit/847ff1e0d94368d94b2e173e4e0908e115568ef3), [`259d409`](https://github.com/mastra-ai/mastra/commit/259d409a514174299dbde1ff5e1121209b3ba850), [`9e16c68`](https://github.com/mastra-ai/mastra/commit/9e16c6818b6485ccb43df28aba6f3a2219d28662), [`cefca33`](https://github.com/mastra-ai/mastra/commit/cefca33ae666e69810c935fedf95a929c173d1d7), [`d00e8c5`](https://github.com/mastra-ai/mastra/commit/d00e8c50daebe5bce5bf2f48bde39c86fc3d2fe4), [`36fa7e2`](https://github.com/mastra-ai/mastra/commit/36fa7e24d14e58a1eb46147097b32f583e5b8775), [`87e9774`](https://github.com/mastra-ai/mastra/commit/87e97741c1e493cd6d62f478eb810b49bda4d57c), [`65a72e7`](https://github.com/mastra-ai/mastra/commit/65a72e70c25eedea8ff985a6624b96be2850236b), [`0f77241`](https://github.com/mastra-ai/mastra/commit/0f7724108806703799a8ba80ad0f09414afd5066), [`92ff509`](https://github.com/mastra-ai/mastra/commit/92ff5098ef8a990438ca038077021a5f7541ec1d), [`3fce5e7`](https://github.com/mastra-ai/mastra/commit/3fce5e70d011d289043e75003ef3336ed4aa43c3), [`a763592`](https://github.com/mastra-ai/mastra/commit/a763592c3db46963ef1011cfe16fe372816e775e), [`80c7737`](https://github.com/mastra-ai/mastra/commit/80c7737e32d7917b5f356957d67c169d01744fd3), [`3f1cf47`](https://github.com/mastra-ai/mastra/commit/3f1cf476f74c1e4cc2df908837e05853a5347e31)]:
33
+ - @mastra/core@1.38.0-alpha.3
34
+ - @mastra/agent-browser@0.3.0-alpha.0
package/LICENSE.md ADDED
@@ -0,0 +1,30 @@
1
+ Portions of this software are licensed as follows:
2
+
3
+ - All content that resides under any directory named "ee/" within this
4
+ repository, including but not limited to:
5
+ - `packages/core/src/auth/ee/`
6
+ - `packages/server/src/server/auth/ee/`
7
+ is licensed under the license defined in `ee/LICENSE`.
8
+
9
+ - All third-party components incorporated into the Mastra Software are
10
+ licensed under the original license provided by the owner of the
11
+ applicable component.
12
+
13
+ - Content outside of the above-mentioned directories or restrictions is
14
+ available under the "Apache License 2.0" as defined below.
15
+
16
+ # Apache License 2.0
17
+
18
+ Copyright (c) 2025 Kepler Software, Inc.
19
+
20
+ Licensed under the Apache License, Version 2.0 (the "License");
21
+ you may not use this file except in compliance with the License.
22
+ You may obtain a copy of the License at
23
+
24
+ http://www.apache.org/licenses/LICENSE-2.0
25
+
26
+ Unless required by applicable law or agreed to in writing, software
27
+ distributed under the License is distributed on an "AS IS" BASIS,
28
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
29
+ See the License for the specific language governing permissions and
30
+ limitations under the License.
package/dist/index.cjs ADDED
@@ -0,0 +1,259 @@
1
+ 'use strict';
2
+
3
+ var agentBrowser = require('@mastra/agent-browser');
4
+ var agentBrowser$1 = require('agent-browser');
5
+ var firecrawl = require('firecrawl');
6
+
7
+ // src/firecrawl-browser.ts
8
+ var FirecrawlAgentBrowserThreadManager = class extends agentBrowser.AgentBrowserThreadManager {
9
+ firecrawl;
10
+ resolveWebSocketUrl;
11
+ sessionOptions;
12
+ constructor(config) {
13
+ super(config);
14
+ this.firecrawl = config.firecrawl;
15
+ this.resolveWebSocketUrl = config.resolveWebSocketUrl;
16
+ this.sessionOptions = config.sessionOptions ?? {};
17
+ }
18
+ async createSession(threadId) {
19
+ const savedState = this.getSavedBrowserState(threadId);
20
+ const session = {
21
+ threadId,
22
+ createdAt: Date.now(),
23
+ browserState: savedState
24
+ };
25
+ if (this.scope === "thread") {
26
+ const createRes = await this.firecrawl.browser({
27
+ ttl: this.sessionOptions.ttl,
28
+ activityTtl: this.sessionOptions.activityTtl,
29
+ streamWebView: this.sessionOptions.streamWebView,
30
+ profile: this.sessionOptions.profile,
31
+ integration: this.sessionOptions.integration,
32
+ origin: this.sessionOptions.origin
33
+ });
34
+ if (!createRes.success || !createRes.id || !createRes.cdpUrl) {
35
+ const msg = createRes.error ?? "Firecrawl browser session creation failed";
36
+ const err = new Error(`Firecrawl browser(): ${msg}`);
37
+ if (createRes.id) {
38
+ try {
39
+ await this.firecrawl.deleteBrowser(createRes.id);
40
+ } catch (cleanupErr) {
41
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);
42
+ }
43
+ }
44
+ throw err;
45
+ }
46
+ session.firecrawlSessionId = createRes.id;
47
+ const manager = new agentBrowser$1.BrowserManager();
48
+ const wsUrl = await this.resolveWebSocketUrl(createRes.cdpUrl);
49
+ const launchOptions = {
50
+ headless: this.browserConfig.headless ?? true,
51
+ viewport: this.browserConfig.viewport,
52
+ profile: this.browserConfig.profile,
53
+ executablePath: this.browserConfig.executablePath,
54
+ storageState: this.browserConfig.storageState,
55
+ cdpUrl: wsUrl
56
+ };
57
+ try {
58
+ await manager.launch(launchOptions);
59
+ } catch (error) {
60
+ try {
61
+ await manager.close();
62
+ } catch {
63
+ }
64
+ try {
65
+ await this.firecrawl.deleteBrowser(createRes.id);
66
+ } catch {
67
+ }
68
+ throw error;
69
+ }
70
+ session.manager = manager;
71
+ this.threadManagers.set(threadId, manager);
72
+ try {
73
+ if (savedState && savedState.tabs.length > 0) {
74
+ this.logger?.debug?.(`Restoring browser state for thread ${threadId}: ${savedState.tabs.length} tabs`);
75
+ await this.restoreBrowserState(manager, savedState);
76
+ }
77
+ this.onBrowserCreated?.(manager, threadId);
78
+ } catch (error) {
79
+ this.threadManagers.delete(threadId);
80
+ session.manager = void 0;
81
+ try {
82
+ await manager.close();
83
+ } catch {
84
+ }
85
+ if (session.firecrawlSessionId) {
86
+ try {
87
+ await this.firecrawl.deleteBrowser(session.firecrawlSessionId);
88
+ } catch {
89
+ }
90
+ }
91
+ throw error;
92
+ }
93
+ }
94
+ return session;
95
+ }
96
+ async doDestroySession(session) {
97
+ if (this.scope === "thread" && session.manager) {
98
+ try {
99
+ await session.manager.close();
100
+ } catch {
101
+ }
102
+ this.threadManagers.delete(session.threadId);
103
+ }
104
+ if (session.firecrawlSessionId) {
105
+ try {
106
+ await this.firecrawl.deleteBrowser(session.firecrawlSessionId);
107
+ } catch {
108
+ }
109
+ }
110
+ }
111
+ };
112
+
113
+ // src/resolve-cdp.ts
114
+ async function resolveCdpWebSocketUrl(url, logger) {
115
+ if (url.startsWith("ws://") || url.startsWith("wss://")) {
116
+ return url;
117
+ }
118
+ if (url.startsWith("http://") || url.startsWith("https://")) {
119
+ const baseUrl = url.replace(/\/$/, "");
120
+ const versionUrl = `${baseUrl}/json/version`;
121
+ logger?.debug?.(`Resolving WebSocket URL from ${versionUrl}`);
122
+ const controller = new AbortController();
123
+ const timeoutId = setTimeout(() => controller.abort(), 1e4);
124
+ try {
125
+ const response = await fetch(versionUrl, { signal: controller.signal });
126
+ if (!response.ok) {
127
+ throw new Error(
128
+ `Failed to fetch CDP version info from ${versionUrl}: ${response.status} ${response.statusText}`
129
+ );
130
+ }
131
+ const data = await response.json();
132
+ if (!data.webSocketDebuggerUrl) {
133
+ throw new Error(`No webSocketDebuggerUrl found in CDP version response from ${versionUrl}`);
134
+ }
135
+ logger?.debug?.(`Resolved WebSocket URL: ${data.webSocketDebuggerUrl}`);
136
+ return data.webSocketDebuggerUrl;
137
+ } catch (error) {
138
+ if (error instanceof Error && error.name === "AbortError") {
139
+ throw new Error(`Timeout resolving WebSocket URL from ${versionUrl} (10s)`);
140
+ }
141
+ throw error;
142
+ } finally {
143
+ clearTimeout(timeoutId);
144
+ }
145
+ }
146
+ return url;
147
+ }
148
+
149
+ // src/firecrawl-browser.ts
150
+ function pickSessionOpts(c) {
151
+ return c.firecrawl ?? {};
152
+ }
153
+ function toBaseConfig(config) {
154
+ const { apiKey: _a, apiUrl: _u, firecrawl: _f, ...rest } = config;
155
+ return rest;
156
+ }
157
+ var FirecrawlBrowser = class extends agentBrowser.AgentBrowser {
158
+ name = "FirecrawlBrowser";
159
+ provider = "firecrawl/browser-sandbox";
160
+ firecrawl;
161
+ sessionOpts;
162
+ sharedFirecrawlSessionId;
163
+ constructor(config) {
164
+ const apiKey = config.apiKey ?? process.env.FIRECRAWL_API_KEY;
165
+ if (!apiKey) {
166
+ throw new Error("FirecrawlBrowser requires `apiKey` or FIRECRAWL_API_KEY");
167
+ }
168
+ const fc = new firecrawl.Firecrawl({ apiKey, apiUrl: config.apiUrl });
169
+ const sessionOpts = pickSessionOpts(config);
170
+ super({
171
+ ...toBaseConfig(config),
172
+ createThreadManager: (opts) => new FirecrawlAgentBrowserThreadManager({
173
+ ...opts,
174
+ firecrawl: fc,
175
+ resolveWebSocketUrl: (url) => resolveCdpWebSocketUrl(url, opts.logger),
176
+ sessionOptions: sessionOpts
177
+ })
178
+ });
179
+ this.firecrawl = fc;
180
+ this.sessionOpts = sessionOpts;
181
+ }
182
+ async doLaunch() {
183
+ const scope = this.threadManager.getScope();
184
+ if (scope === "thread") {
185
+ await super.doLaunch();
186
+ return;
187
+ }
188
+ const createRes = await this.firecrawl.browser({
189
+ ttl: this.sessionOpts.ttl,
190
+ activityTtl: this.sessionOpts.activityTtl,
191
+ streamWebView: this.sessionOpts.streamWebView,
192
+ profile: this.sessionOpts.profile,
193
+ integration: this.sessionOpts.integration,
194
+ origin: this.sessionOpts.origin
195
+ });
196
+ if (!createRes.success || !createRes.id || !createRes.cdpUrl) {
197
+ const msg = createRes.error ?? "Firecrawl browser session creation failed";
198
+ const err = new Error(`Firecrawl browser(): ${msg}`);
199
+ if (createRes.id) {
200
+ try {
201
+ await this.firecrawl.deleteBrowser(createRes.id);
202
+ } catch (cleanupErr) {
203
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);
204
+ }
205
+ }
206
+ throw err;
207
+ }
208
+ const sessionId = createRes.id;
209
+ this.sharedManager = new agentBrowser$1.BrowserManager();
210
+ try {
211
+ const localConfig = this.config;
212
+ const wsUrl = await resolveCdpWebSocketUrl(createRes.cdpUrl, this.logger);
213
+ const launchOptions = {
214
+ headless: localConfig.headless ?? true,
215
+ viewport: localConfig.viewport,
216
+ profile: localConfig.profile,
217
+ executablePath: localConfig.executablePath,
218
+ storageState: localConfig.storageState,
219
+ cdpUrl: wsUrl
220
+ };
221
+ await this.sharedManager.launch(launchOptions);
222
+ this.threadManager.setSharedManager(this.sharedManager);
223
+ this.setupCloseListenerForSharedScope(this.sharedManager);
224
+ this.sharedFirecrawlSessionId = sessionId;
225
+ } catch (launchErr) {
226
+ try {
227
+ await this.sharedManager.close();
228
+ } catch (closeErr) {
229
+ this.logger?.warn?.(`BrowserManager.close() after failed shared launch: ${closeErr}`);
230
+ }
231
+ try {
232
+ await this.firecrawl.deleteBrowser(sessionId);
233
+ } catch (delErr) {
234
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${sessionId}) after failed shared launch: ${delErr}`);
235
+ }
236
+ this.sharedManager = null;
237
+ this.sharedFirecrawlSessionId = void 0;
238
+ throw launchErr;
239
+ }
240
+ }
241
+ async doClose() {
242
+ const sid = this.sharedFirecrawlSessionId;
243
+ await super.doClose();
244
+ if (sid) {
245
+ try {
246
+ await this.firecrawl.deleteBrowser(sid);
247
+ } catch (err) {
248
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${sid}) failed: ${err}`);
249
+ }
250
+ this.sharedFirecrawlSessionId = void 0;
251
+ }
252
+ }
253
+ };
254
+
255
+ exports.FirecrawlAgentBrowserThreadManager = FirecrawlAgentBrowserThreadManager;
256
+ exports.FirecrawlBrowser = FirecrawlBrowser;
257
+ exports.resolveCdpWebSocketUrl = resolveCdpWebSocketUrl;
258
+ //# sourceMappingURL=index.cjs.map
259
+ //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/firecrawl-thread-manager.ts","../src/resolve-cdp.ts","../src/firecrawl-browser.ts"],"names":["AgentBrowserThreadManager","BrowserManager","AgentBrowser","Firecrawl"],"mappings":";;;;;;;AAyBO,IAAM,kCAAA,GAAN,cAAiDA,sCAAA,CAA0B;AAAA,EAC/D,SAAA;AAAA,EACA,mBAAA;AAAA,EACA,cAAA;AAAA,EAEjB,YAAY,MAAA,EAAkD;AAC5D,IAAA,KAAA,CAAM,MAAM,CAAA;AACZ,IAAA,IAAA,CAAK,YAAY,MAAA,CAAO,SAAA;AACxB,IAAA,IAAA,CAAK,sBAAsB,MAAA,CAAO,mBAAA;AAClC,IAAA,IAAA,CAAK,cAAA,GAAiB,MAAA,CAAO,cAAA,IAAkB,EAAC;AAAA,EAClD;AAAA,EAEA,MAAyB,cAAc,QAAA,EAAyD;AAC9F,IAAA,MAAM,UAAA,GAAa,IAAA,CAAK,oBAAA,CAAqB,QAAQ,CAAA;AAErD,IAAA,MAAM,OAAA,GAAwC;AAAA,MAC5C,QAAA;AAAA,MACA,SAAA,EAAW,KAAK,GAAA,EAAI;AAAA,MACpB,YAAA,EAAc;AAAA,KAChB;AAEA,IAAA,IAAI,IAAA,CAAK,UAAU,QAAA,EAAU;AAC3B,MAAA,MAAM,SAAA,GAAY,MAAM,IAAA,CAAK,SAAA,CAAU,OAAA,CAAQ;AAAA,QAC7C,GAAA,EAAK,KAAK,cAAA,CAAe,GAAA;AAAA,QACzB,WAAA,EAAa,KAAK,cAAA,CAAe,WAAA;AAAA,QACjC,aAAA,EAAe,KAAK,cAAA,CAAe,aAAA;AAAA,QACnC,OAAA,EAAS,KAAK,cAAA,CAAe,OAAA;AAAA,QAC7B,WAAA,EAAa,KAAK,cAAA,CAAe,WAAA;AAAA,QACjC,MAAA,EAAQ,KAAK,cAAA,CAAe;AAAA,OAC7B,CAAA;AAED,MAAA,IAAI,CAAC,UAAU,OAAA,IAAW,CAAC,UAAU,EAAA,IAAM,CAAC,UAAU,MAAA,EAAQ;AAC5D,QAAA,MAAM,GAAA,GAAM,UAAU,KAAA,IAAS,2CAAA;AAC/B,QAAA,MAAM,GAAA,GAAM,IAAI,KAAA,CAAM,CAAA,qBAAA,EAAwB,GAAG,CAAA,CAAE,CAAA;AACnD,QAAA,IAAI,UAAU,EAAA,EAAI;AAChB,UAAA,IAAI;AACF,YAAA,MAAM,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,SAAA,CAAU,EAAE,CAAA;AAAA,UACjD,SAAS,UAAA,EAAY;AACnB,YAAA,IAAA,CAAK,QAAQ,IAAA,GAAO,CAAA,wBAAA,EAA2B,UAAU,EAAE,CAAA,0BAAA,EAA6B,UAAU,CAAA,CAAE,CAAA;AAAA,UACtG;AAAA,QACF;AACA,QAAA,MAAM,GAAA;AAAA,MACR;AAEA,MAAA,OAAA,CAAQ,qBAAqB,SAAA,CAAU,EAAA;AAEvC,MAAA,MAAM,OAAA,GAAU,IAAIC,6BAAA,EAAe;AAEnC,MAAA,MAAM,KAAA,GAAQ,MAAM,IAAA,CAAK,mBAAA,CAAoB,UAAU,MAAM,CAAA;AAE7D,MAAA,MAAM,aAAA,GAAsC;AAAA,QAC1C,QAAA,EAAU,IAAA,CAAK,aAAA,CAAc,QAAA,IAAY,IAAA;AAAA,QACzC,QAAA,EAAU,KAAK,aAAA,CAAc,QAAA;AAAA,QAC7B,OAAA,EAAS,KAAK,aAAA,CAAc,OAAA;AAAA,QAC5B,cAAA,EAAgB,KAAK,aAAA,CAAc,cAAA;AAAA,QACnC,YAAA,EAAc,KAAK,aAAA,CAAc,YAAA;AAAA,QACjC,MAAA,EAAQ;AAAA,OACV;AAEA,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,CAAQ,OAAO,aAAa,CAAA;AAAA,MACpC,SAAS,KAAA,EAAO;AACd,QAAA,IAAI;AACF,UAAA,MAAM,QAAQ,KAAA,EAAM;AAAA,QACtB,CAAA,CAAA,MAAQ;AAAA,QAER;AACA,QAAA,IAAI;AACF,UAAA,MAAM,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,SAAA,CAAU,EAAE,CAAA;AAAA,QACjD,CAAA,CAAA,MAAQ;AAAA,QAER;AACA,QAAA,MAAM,KAAA;AAAA,MACR;AAEA,MAAA,OAAA,CAAQ,OAAA,GAAU,OAAA;AAClB,MAAA,IAAA,CAAK,cAAA,CAAe,GAAA,CAAI,QAAA,EAAU,OAAO,CAAA;AAEzC,MAAA,IAAI;AACF,QAAA,IAAI,UAAA,IAAc,UAAA,CAAW,IAAA,CAAK,MAAA,GAAS,CAAA,EAAG;AAC5C,UAAA,IAAA,CAAK,MAAA,EAAQ,QAAQ,CAAA,mCAAA,EAAsC,QAAQ,KAAK,UAAA,CAAW,IAAA,CAAK,MAAM,CAAA,KAAA,CAAO,CAAA;AACrG,UAAA,MAAM,IAAA,CAAK,mBAAA,CAAoB,OAAA,EAAS,UAAU,CAAA;AAAA,QACpD;AACA,QAAA,IAAA,CAAK,gBAAA,GAAmB,SAAS,QAAQ,CAAA;AAAA,MAC3C,SAAS,KAAA,EAAO;AACd,QAAA,IAAA,CAAK,cAAA,CAAe,OAAO,QAAQ,CAAA;AACnC,QAAA,OAAA,CAAQ,OAAA,GAAU,MAAA;AAClB,QAAA,IAAI;AACF,UAAA,MAAM,QAAQ,KAAA,EAAM;AAAA,QACtB,CAAA,CAAA,MAAQ;AAAA,QAER;AACA,QAAA,IAAI,QAAQ,kBAAA,EAAoB;AAC9B,UAAA,IAAI;AACF,YAAA,MAAM,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,OAAA,CAAQ,kBAAkB,CAAA;AAAA,UAC/D,CAAA,CAAA,MAAQ;AAAA,UAER;AAAA,QACF;AACA,QAAA,MAAM,KAAA;AAAA,MACR;AAAA,IACF;AAEA,IAAA,OAAO,OAAA;AAAA,EACT;AAAA,EAEA,MAAyB,iBAAiB,OAAA,EAAsD;AAC9F,IAAA,IAAI,IAAA,CAAK,KAAA,KAAU,QAAA,IAAY,OAAA,CAAQ,OAAA,EAAS;AAC9C,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,CAAQ,QAAQ,KAAA,EAAM;AAAA,MAC9B,CAAA,CAAA,MAAQ;AAAA,MAER;AACA,MAAA,IAAA,CAAK,cAAA,CAAe,MAAA,CAAO,OAAA,CAAQ,QAAQ,CAAA;AAAA,IAC7C;AAEA,IAAA,IAAI,QAAQ,kBAAA,EAAoB;AAC9B,MAAA,IAAI;AACF,QAAA,MAAM,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,OAAA,CAAQ,kBAAkB,CAAA;AAAA,MAC/D,CAAA,CAAA,MAAQ;AAAA,MAER;AAAA,IACF;AAAA,EACF;AACF;;;AClJA,eAAsB,sBAAA,CACpB,KACA,MAAA,EACiB;AACjB,EAAA,IAAI,IAAI,UAAA,CAAW,OAAO,KAAK,GAAA,CAAI,UAAA,CAAW,QAAQ,CAAA,EAAG;AACvD,IAAA,OAAO,GAAA;AAAA,EACT;AAEA,EAAA,IAAI,IAAI,UAAA,CAAW,SAAS,KAAK,GAAA,CAAI,UAAA,CAAW,UAAU,CAAA,EAAG;AAC3D,IAAA,MAAM,OAAA,GAAU,GAAA,CAAI,OAAA,CAAQ,KAAA,EAAO,EAAE,CAAA;AACrC,IAAA,MAAM,UAAA,GAAa,GAAG,OAAO,CAAA,aAAA,CAAA;AAE7B,IAAA,MAAA,EAAQ,KAAA,GAAQ,CAAA,6BAAA,EAAgC,UAAU,CAAA,CAAE,CAAA;AAE5D,IAAA,MAAM,UAAA,GAAa,IAAI,eAAA,EAAgB;AACvC,IAAA,MAAM,YAAY,UAAA,CAAW,MAAM,UAAA,CAAW,KAAA,IAAS,GAAK,CAAA;AAE5D,IAAA,IAAI;AACF,MAAA,MAAM,QAAA,GAAW,MAAM,KAAA,CAAM,UAAA,EAAY,EAAE,MAAA,EAAQ,UAAA,CAAW,QAAQ,CAAA;AAEtE,MAAA,IAAI,CAAC,SAAS,EAAA,EAAI;AAChB,QAAA,MAAM,IAAI,KAAA;AAAA,UACR,yCAAyC,UAAU,CAAA,EAAA,EAAK,SAAS,MAAM,CAAA,CAAA,EAAI,SAAS,UAAU,CAAA;AAAA,SAChG;AAAA,MACF;AAEA,MAAA,MAAM,IAAA,GAAQ,MAAM,QAAA,CAAS,IAAA,EAAK;AAClC,MAAA,IAAI,CAAC,KAAK,oBAAA,EAAsB;AAC9B,QAAA,MAAM,IAAI,KAAA,CAAM,CAAA,2DAAA,EAA8D,UAAU,CAAA,CAAE,CAAA;AAAA,MAC5F;AAEA,MAAA,MAAA,EAAQ,KAAA,GAAQ,CAAA,wBAAA,EAA2B,IAAA,CAAK,oBAAoB,CAAA,CAAE,CAAA;AACtE,MAAA,OAAO,IAAA,CAAK,oBAAA;AAAA,IACd,SAAS,KAAA,EAAO;AACd,MAAA,IAAI,KAAA,YAAiB,KAAA,IAAS,KAAA,CAAM,IAAA,KAAS,YAAA,EAAc;AACzD,QAAA,MAAM,IAAI,KAAA,CAAM,CAAA,qCAAA,EAAwC,UAAU,CAAA,MAAA,CAAQ,CAAA;AAAA,MAC5E;AACA,MAAA,MAAM,KAAA;AAAA,IACR,CAAA,SAAE;AACA,MAAA,YAAA,CAAa,SAAS,CAAA;AAAA,IACxB;AAAA,EACF;AAEA,EAAA,OAAO,GAAA;AACT;;;ACtCA,SAAS,gBAAgB,CAAA,EAA2D;AAClF,EAAA,OAAO,CAAA,CAAE,aAAa,EAAC;AACzB;AAEA,SAAS,aAAa,MAAA,EAAoD;AACxE,EAAA,MAAM,EAAE,QAAQ,EAAA,EAAI,MAAA,EAAQ,IAAI,SAAA,EAAW,EAAA,EAAI,GAAG,IAAA,EAAK,GAAI,MAAA;AAC3D,EAAA,OAAO,IAAA;AACT;AAMO,IAAM,gBAAA,GAAN,cAA+BC,yBAAA,CAAa;AAAA,EAC/B,IAAA,GAAO,kBAAA;AAAA,EACP,QAAA,GAAW,2BAAA;AAAA,EAKZ,SAAA;AAAA,EACA,WAAA;AAAA,EACT,wBAAA;AAAA,EAER,YAAY,MAAA,EAAgC;AAC1C,IAAA,MAAM,MAAA,GAAS,MAAA,CAAO,MAAA,IAAU,OAAA,CAAQ,GAAA,CAAI,iBAAA;AAC5C,IAAA,IAAI,CAAC,MAAA,EAAQ;AACX,MAAA,MAAM,IAAI,MAAM,yDAAyD,CAAA;AAAA,IAC3E;AACA,IAAA,MAAM,EAAA,GAAK,IAAIC,mBAAA,CAAU,EAAE,QAAQ,MAAA,EAAQ,MAAA,CAAO,QAAQ,CAAA;AAC1D,IAAA,MAAM,WAAA,GAAc,gBAAgB,MAAM,CAAA;AAE1C,IAAA,KAAA,CAAM;AAAA,MACJ,GAAG,aAAa,MAAM,CAAA;AAAA,MACtB,mBAAA,EAAqB,CAAA,IAAA,KACnB,IAAI,kCAAA,CAAmC;AAAA,QACrC,GAAG,IAAA;AAAA,QACH,SAAA,EAAW,EAAA;AAAA,QACX,mBAAA,EAAqB,CAAA,GAAA,KAAO,sBAAA,CAAuB,GAAA,EAAK,KAAK,MAAM,CAAA;AAAA,QACnE,cAAA,EAAgB;AAAA,OACjB;AAAA,KACJ,CAAA;AACD,IAAA,IAAA,CAAK,SAAA,GAAY,EAAA;AACjB,IAAA,IAAA,CAAK,WAAA,GAAc,WAAA;AAAA,EACrB;AAAA,EAEA,MAAyB,QAAA,GAA0B;AACjD,IAAA,MAAM,KAAA,GAAQ,IAAA,CAAK,aAAA,CAAc,QAAA,EAAS;AAC1C,IAAA,IAAI,UAAU,QAAA,EAAU;AACtB,MAAA,MAAM,MAAM,QAAA,EAAS;AACrB,MAAA;AAAA,IACF;AAEA,IAAA,MAAM,SAAA,GAAY,MAAM,IAAA,CAAK,SAAA,CAAU,OAAA,CAAQ;AAAA,MAC7C,GAAA,EAAK,KAAK,WAAA,CAAY,GAAA;AAAA,MACtB,WAAA,EAAa,KAAK,WAAA,CAAY,WAAA;AAAA,MAC9B,aAAA,EAAe,KAAK,WAAA,CAAY,aAAA;AAAA,MAChC,OAAA,EAAS,KAAK,WAAA,CAAY,OAAA;AAAA,MAC1B,WAAA,EAAa,KAAK,WAAA,CAAY,WAAA;AAAA,MAC9B,MAAA,EAAQ,KAAK,WAAA,CAAY;AAAA,KAC1B,CAAA;AAED,IAAA,IAAI,CAAC,UAAU,OAAA,IAAW,CAAC,UAAU,EAAA,IAAM,CAAC,UAAU,MAAA,EAAQ;AAC5D,MAAA,MAAM,GAAA,GAAM,UAAU,KAAA,IAAS,2CAAA;AAC/B,MAAA,MAAM,GAAA,GAAM,IAAI,KAAA,CAAM,CAAA,qBAAA,EAAwB,GAAG,CAAA,CAAE,CAAA;AACnD,MAAA,IAAI,UAAU,EAAA,EAAI;AAChB,QAAA,IAAI;AACF,UAAA,MAAM,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,SAAA,CAAU,EAAE,CAAA;AAAA,QACjD,SAAS,UAAA,EAAY;AACnB,UAAA,IAAA,CAAK,QAAQ,IAAA,GAAO,CAAA,wBAAA,EAA2B,UAAU,EAAE,CAAA,0BAAA,EAA6B,UAAU,CAAA,CAAE,CAAA;AAAA,QACtG;AAAA,MACF;AACA,MAAA,MAAM,GAAA;AAAA,IACR;AAEA,IAAA,MAAM,YAAY,SAAA,CAAU,EAAA;AAC5B,IAAA,IAAA,CAAK,aAAA,GAAgB,IAAIF,6BAAAA,EAAe;AAExC,IAAA,IAAI;AACF,MAAA,MAAM,cAAc,IAAA,CAAK,MAAA;AACzB,MAAA,MAAM,QAAQ,MAAM,sBAAA,CAAuB,SAAA,CAAU,MAAA,EAAQ,KAAK,MAAM,CAAA;AAExE,MAAA,MAAM,aAAA,GAAsC;AAAA,QAC1C,QAAA,EAAU,YAAY,QAAA,IAAY,IAAA;AAAA,QAClC,UAAU,WAAA,CAAY,QAAA;AAAA,QACtB,SAAS,WAAA,CAAY,OAAA;AAAA,QACrB,gBAAgB,WAAA,CAAY,cAAA;AAAA,QAC5B,cAAc,WAAA,CAAY,YAAA;AAAA,QAC1B,MAAA,EAAQ;AAAA,OACV;AAEA,MAAA,MAAM,IAAA,CAAK,aAAA,CAAc,MAAA,CAAO,aAAa,CAAA;AAC7C,MAAA,IAAA,CAAK,aAAA,CAAc,gBAAA,CAAiB,IAAA,CAAK,aAAa,CAAA;AACtD,MAAA,IAAA,CAAK,gCAAA,CAAiC,KAAK,aAAa,CAAA;AACxD,MAAA,IAAA,CAAK,wBAAA,GAA2B,SAAA;AAAA,IAClC,SAAS,SAAA,EAAW;AAClB,MAAA,IAAI;AACF,QAAA,MAAM,IAAA,CAAK,cAAc,KAAA,EAAM;AAAA,MACjC,SAAS,QAAA,EAAU;AACjB,QAAA,IAAA,CAAK,MAAA,EAAQ,IAAA,GAAO,CAAA,mDAAA,EAAsD,QAAQ,CAAA,CAAE,CAAA;AAAA,MACtF;AACA,MAAA,IAAI;AACF,QAAA,MAAM,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,SAAS,CAAA;AAAA,MAC9C,SAAS,MAAA,EAAQ;AACf,QAAA,IAAA,CAAK,QAAQ,IAAA,GAAO,CAAA,wBAAA,EAA2B,SAAS,CAAA,8BAAA,EAAiC,MAAM,CAAA,CAAE,CAAA;AAAA,MACnG;AACA,MAAA,IAAA,CAAK,aAAA,GAAgB,IAAA;AACrB,MAAA,IAAA,CAAK,wBAAA,GAA2B,MAAA;AAChC,MAAA,MAAM,SAAA;AAAA,IACR;AAAA,EACF;AAAA,EAEA,MAAyB,OAAA,GAAyB;AAChD,IAAA,MAAM,MAAM,IAAA,CAAK,wBAAA;AACjB,IAAA,MAAM,MAAM,OAAA,EAAQ;AACpB,IAAA,IAAI,GAAA,EAAK;AACP,MAAA,IAAI;AACF,QAAA,MAAM,IAAA,CAAK,SAAA,CAAU,aAAA,CAAc,GAAG,CAAA;AAAA,MACxC,SAAS,GAAA,EAAK;AACZ,QAAA,IAAA,CAAK,QAAQ,IAAA,GAAO,CAAA,wBAAA,EAA2B,GAAG,CAAA,UAAA,EAAa,GAAG,CAAA,CAAE,CAAA;AAAA,MACtE;AACA,MAAA,IAAA,CAAK,wBAAA,GAA2B,MAAA;AAAA,IAClC;AAAA,EACF;AACF","file":"index.cjs","sourcesContent":["import type { AgentBrowserSession, AgentBrowserThreadManagerConfig } from '@mastra/agent-browser';\nimport { AgentBrowserThreadManager } from '@mastra/agent-browser';\nimport type { BrowserLaunchOptions } from 'agent-browser';\nimport { BrowserManager } from 'agent-browser';\nimport type { Firecrawl } from 'firecrawl';\nimport type { FirecrawlBrowserSessionOptions } from './types';\n\n/**\n * Thread session with Firecrawl sandbox id for cleanup.\n */\nexport interface FirecrawlAgentBrowserSession extends AgentBrowserSession {\n firecrawlSessionId?: string;\n}\n\nexport interface FirecrawlAgentBrowserThreadManagerConfig extends AgentBrowserThreadManagerConfig {\n firecrawl: Firecrawl;\n /** Resolve HTTP CDP URL to WebSocket URL. */\n resolveWebSocketUrl: (url: string) => Promise<string>;\n /** Options for each `firecrawl.browser()` call (thread scope = one call per thread). */\n sessionOptions?: FirecrawlBrowserSessionOptions;\n}\n\n/**\n * Provisions a dedicated Firecrawl Browser Sandbox session per Mastra thread and connects Playwright over CDP.\n */\nexport class FirecrawlAgentBrowserThreadManager extends AgentBrowserThreadManager {\n private readonly firecrawl: Firecrawl;\n private readonly resolveWebSocketUrl: (url: string) => Promise<string>;\n private readonly sessionOptions: FirecrawlBrowserSessionOptions;\n\n constructor(config: FirecrawlAgentBrowserThreadManagerConfig) {\n super(config);\n this.firecrawl = config.firecrawl;\n this.resolveWebSocketUrl = config.resolveWebSocketUrl;\n this.sessionOptions = config.sessionOptions ?? {};\n }\n\n protected override async createSession(threadId: string): Promise<FirecrawlAgentBrowserSession> {\n const savedState = this.getSavedBrowserState(threadId);\n\n const session: FirecrawlAgentBrowserSession = {\n threadId,\n createdAt: Date.now(),\n browserState: savedState,\n };\n\n if (this.scope === 'thread') {\n const createRes = await this.firecrawl.browser({\n ttl: this.sessionOptions.ttl,\n activityTtl: this.sessionOptions.activityTtl,\n streamWebView: this.sessionOptions.streamWebView,\n profile: this.sessionOptions.profile,\n integration: this.sessionOptions.integration,\n origin: this.sessionOptions.origin,\n });\n\n if (!createRes.success || !createRes.id || !createRes.cdpUrl) {\n const msg = createRes.error ?? 'Firecrawl browser session creation failed';\n const err = new Error(`Firecrawl browser(): ${msg}`);\n if (createRes.id) {\n try {\n await this.firecrawl.deleteBrowser(createRes.id);\n } catch (cleanupErr) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);\n }\n }\n throw err;\n }\n\n session.firecrawlSessionId = createRes.id;\n\n const manager = new BrowserManager();\n\n const wsUrl = await this.resolveWebSocketUrl(createRes.cdpUrl);\n\n const launchOptions: BrowserLaunchOptions = {\n headless: this.browserConfig.headless ?? true,\n viewport: this.browserConfig.viewport,\n profile: this.browserConfig.profile,\n executablePath: this.browserConfig.executablePath,\n storageState: this.browserConfig.storageState,\n cdpUrl: wsUrl,\n };\n\n try {\n await manager.launch(launchOptions);\n } catch (error) {\n try {\n await manager.close();\n } catch {\n // ignore\n }\n try {\n await this.firecrawl.deleteBrowser(createRes.id);\n } catch {\n // ignore\n }\n throw error;\n }\n\n session.manager = manager;\n this.threadManagers.set(threadId, manager);\n\n try {\n if (savedState && savedState.tabs.length > 0) {\n this.logger?.debug?.(`Restoring browser state for thread ${threadId}: ${savedState.tabs.length} tabs`);\n await this.restoreBrowserState(manager, savedState);\n }\n this.onBrowserCreated?.(manager, threadId);\n } catch (error) {\n this.threadManagers.delete(threadId);\n session.manager = undefined;\n try {\n await manager.close();\n } catch {\n // ignore\n }\n if (session.firecrawlSessionId) {\n try {\n await this.firecrawl.deleteBrowser(session.firecrawlSessionId);\n } catch {\n // ignore\n }\n }\n throw error;\n }\n }\n\n return session;\n }\n\n protected override async doDestroySession(session: FirecrawlAgentBrowserSession): Promise<void> {\n if (this.scope === 'thread' && session.manager) {\n try {\n await session.manager.close();\n } catch {\n // ignore\n }\n this.threadManagers.delete(session.threadId);\n }\n\n if (session.firecrawlSessionId) {\n try {\n await this.firecrawl.deleteBrowser(session.firecrawlSessionId);\n } catch {\n // ignore\n }\n }\n }\n}\n","/**\n * Resolve an HTTP CDP endpoint to a WebSocket URL (same behavior as MastraBrowser.resolveWebSocketUrl).\n */\nexport async function resolveCdpWebSocketUrl(\n url: string,\n logger?: { debug?: (message: string) => void; warn?: (message: string) => void },\n): Promise<string> {\n if (url.startsWith('ws://') || url.startsWith('wss://')) {\n return url;\n }\n\n if (url.startsWith('http://') || url.startsWith('https://')) {\n const baseUrl = url.replace(/\\/$/, '');\n const versionUrl = `${baseUrl}/json/version`;\n\n logger?.debug?.(`Resolving WebSocket URL from ${versionUrl}`);\n\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), 10000);\n\n try {\n const response = await fetch(versionUrl, { signal: controller.signal });\n\n if (!response.ok) {\n throw new Error(\n `Failed to fetch CDP version info from ${versionUrl}: ${response.status} ${response.statusText}`,\n );\n }\n\n const data = (await response.json()) as { webSocketDebuggerUrl?: string };\n if (!data.webSocketDebuggerUrl) {\n throw new Error(`No webSocketDebuggerUrl found in CDP version response from ${versionUrl}`);\n }\n\n logger?.debug?.(`Resolved WebSocket URL: ${data.webSocketDebuggerUrl}`);\n return data.webSocketDebuggerUrl;\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n throw new Error(`Timeout resolving WebSocket URL from ${versionUrl} (10s)`);\n }\n throw error;\n } finally {\n clearTimeout(timeoutId);\n }\n }\n\n return url;\n}\n","import type { AgentBrowserConfig } from '@mastra/agent-browser';\nimport { AgentBrowser } from '@mastra/agent-browser';\nimport type { BrowserLaunchOptions } from 'agent-browser';\nimport { BrowserManager } from 'agent-browser';\nimport { Firecrawl } from 'firecrawl';\nimport { FirecrawlAgentBrowserThreadManager } from './firecrawl-thread-manager';\nimport { resolveCdpWebSocketUrl } from './resolve-cdp';\nimport type { FirecrawlBrowserConfig, FirecrawlBrowserSessionOptions } from './types';\n\nfunction pickSessionOpts(c: FirecrawlBrowserConfig): FirecrawlBrowserSessionOptions {\n return c.firecrawl ?? {};\n}\n\nfunction toBaseConfig(config: FirecrawlBrowserConfig): AgentBrowserConfig {\n const { apiKey: _a, apiUrl: _u, firecrawl: _f, ...rest } = config;\n return rest;\n}\n\n/**\n * Mastra browser provider backed by [Firecrawl Browser Sandbox](https://docs.firecrawl.dev/features/browser):\n * provisions remote sessions via API and drives them with the same deterministic tools as {@link AgentBrowser}.\n */\nexport class FirecrawlBrowser extends AgentBrowser {\n override readonly name = 'FirecrawlBrowser';\n override readonly provider = 'firecrawl/browser-sandbox';\n\n /** Narrowed from base `MastraBrowser` (`unknown`) — same pattern as {@link AgentBrowser}. */\n declare protected sharedManager: BrowserManager | null;\n\n private readonly firecrawl: Firecrawl;\n private readonly sessionOpts: FirecrawlBrowserSessionOptions;\n private sharedFirecrawlSessionId?: string;\n\n constructor(config: FirecrawlBrowserConfig) {\n const apiKey = config.apiKey ?? process.env.FIRECRAWL_API_KEY;\n if (!apiKey) {\n throw new Error('FirecrawlBrowser requires `apiKey` or FIRECRAWL_API_KEY');\n }\n const fc = new Firecrawl({ apiKey, apiUrl: config.apiUrl });\n const sessionOpts = pickSessionOpts(config);\n\n super({\n ...toBaseConfig(config),\n createThreadManager: opts =>\n new FirecrawlAgentBrowserThreadManager({\n ...opts,\n firecrawl: fc,\n resolveWebSocketUrl: url => resolveCdpWebSocketUrl(url, opts.logger),\n sessionOptions: sessionOpts,\n }),\n });\n this.firecrawl = fc;\n this.sessionOpts = sessionOpts;\n }\n\n protected override async doLaunch(): Promise<void> {\n const scope = this.threadManager.getScope();\n if (scope === 'thread') {\n await super.doLaunch();\n return;\n }\n\n const createRes = await this.firecrawl.browser({\n ttl: this.sessionOpts.ttl,\n activityTtl: this.sessionOpts.activityTtl,\n streamWebView: this.sessionOpts.streamWebView,\n profile: this.sessionOpts.profile,\n integration: this.sessionOpts.integration,\n origin: this.sessionOpts.origin,\n });\n\n if (!createRes.success || !createRes.id || !createRes.cdpUrl) {\n const msg = createRes.error ?? 'Firecrawl browser session creation failed';\n const err = new Error(`Firecrawl browser(): ${msg}`);\n if (createRes.id) {\n try {\n await this.firecrawl.deleteBrowser(createRes.id);\n } catch (cleanupErr) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);\n }\n }\n throw err;\n }\n\n const sessionId = createRes.id;\n this.sharedManager = new BrowserManager();\n\n try {\n const localConfig = this.config as AgentBrowserConfig;\n const wsUrl = await resolveCdpWebSocketUrl(createRes.cdpUrl, this.logger);\n\n const launchOptions: BrowserLaunchOptions = {\n headless: localConfig.headless ?? true,\n viewport: localConfig.viewport,\n profile: localConfig.profile,\n executablePath: localConfig.executablePath,\n storageState: localConfig.storageState,\n cdpUrl: wsUrl,\n };\n\n await this.sharedManager.launch(launchOptions);\n this.threadManager.setSharedManager(this.sharedManager);\n this.setupCloseListenerForSharedScope(this.sharedManager);\n this.sharedFirecrawlSessionId = sessionId;\n } catch (launchErr) {\n try {\n await this.sharedManager.close();\n } catch (closeErr) {\n this.logger?.warn?.(`BrowserManager.close() after failed shared launch: ${closeErr}`);\n }\n try {\n await this.firecrawl.deleteBrowser(sessionId);\n } catch (delErr) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${sessionId}) after failed shared launch: ${delErr}`);\n }\n this.sharedManager = null;\n this.sharedFirecrawlSessionId = undefined;\n throw launchErr;\n }\n }\n\n protected override async doClose(): Promise<void> {\n const sid = this.sharedFirecrawlSessionId;\n await super.doClose();\n if (sid) {\n try {\n await this.firecrawl.deleteBrowser(sid);\n } catch (err) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${sid}) failed: ${err}`);\n }\n this.sharedFirecrawlSessionId = undefined;\n }\n }\n}\n"]}
@@ -0,0 +1,94 @@
1
+ import { AgentBrowserConfig, AgentBrowser, AgentBrowserSession, AgentBrowserThreadManager, AgentBrowserThreadManagerConfig } from '@mastra/agent-browser';
2
+ import { BrowserManager } from 'agent-browser';
3
+ import { Firecrawl } from 'firecrawl';
4
+
5
+ /**
6
+ * Options passed to Firecrawl `POST /v2/browser` (see Firecrawl JS SDK `browser()`).
7
+ *
8
+ * **Note:** {@link AgentBrowserConfig.profile} is the Playwright / agent-browser user-data directory
9
+ * (filesystem path). The optional `profile` below is Firecrawl’s **named sandbox profile** for the
10
+ * hosted browser API only—different meaning, same nested key name as required by the SDK.
11
+ */
12
+ interface FirecrawlBrowserSessionOptions {
13
+ /** Max session wall-clock lifetime (seconds, Firecrawl API). */
14
+ ttl?: number;
15
+ /** Idle timeout before the sandbox recycles the session (seconds, Firecrawl API). */
16
+ activityTtl?: number;
17
+ /** When true, Firecrawl may stream WebView frames for the remote session. */
18
+ streamWebView?: boolean;
19
+ /** Firecrawl named profile (not the same as top-level `AgentBrowserConfig.profile`). */
20
+ profile?: {
21
+ /** Saved profile name in Firecrawl. */
22
+ name: string;
23
+ /** Persist profile changes when the session ends. */
24
+ saveChanges?: boolean;
25
+ };
26
+ /** Optional integration label for Firecrawl analytics / routing. */
27
+ integration?: string;
28
+ /** Optional origin hint for the Firecrawl browser session. */
29
+ origin?: string;
30
+ }
31
+ /** Configuration for {@link FirecrawlBrowser}. */
32
+ type FirecrawlBrowserConfig = AgentBrowserConfig & {
33
+ /** Firecrawl API key (or set `FIRECRAWL_API_KEY` in the environment and omit). */
34
+ apiKey?: string;
35
+ /** Base URL for a self-hosted Firecrawl API. */
36
+ apiUrl?: string;
37
+ /**
38
+ * Firecrawl-only session options (`browser()`). Distinct from {@link AgentBrowserConfig.profile}
39
+ * (local Playwright profile path): see {@link FirecrawlBrowserSessionOptions.profile}.
40
+ */
41
+ firecrawl?: FirecrawlBrowserSessionOptions;
42
+ };
43
+
44
+ /**
45
+ * Mastra browser provider backed by [Firecrawl Browser Sandbox](https://docs.firecrawl.dev/features/browser):
46
+ * provisions remote sessions via API and drives them with the same deterministic tools as {@link AgentBrowser}.
47
+ */
48
+ declare class FirecrawlBrowser extends AgentBrowser {
49
+ readonly name = "FirecrawlBrowser";
50
+ readonly provider = "firecrawl/browser-sandbox";
51
+ /** Narrowed from base `MastraBrowser` (`unknown`) — same pattern as {@link AgentBrowser}. */
52
+ protected sharedManager: BrowserManager | null;
53
+ private readonly firecrawl;
54
+ private readonly sessionOpts;
55
+ private sharedFirecrawlSessionId?;
56
+ constructor(config: FirecrawlBrowserConfig);
57
+ protected doLaunch(): Promise<void>;
58
+ protected doClose(): Promise<void>;
59
+ }
60
+
61
+ /**
62
+ * Thread session with Firecrawl sandbox id for cleanup.
63
+ */
64
+ interface FirecrawlAgentBrowserSession extends AgentBrowserSession {
65
+ firecrawlSessionId?: string;
66
+ }
67
+ interface FirecrawlAgentBrowserThreadManagerConfig extends AgentBrowserThreadManagerConfig {
68
+ firecrawl: Firecrawl;
69
+ /** Resolve HTTP CDP URL to WebSocket URL. */
70
+ resolveWebSocketUrl: (url: string) => Promise<string>;
71
+ /** Options for each `firecrawl.browser()` call (thread scope = one call per thread). */
72
+ sessionOptions?: FirecrawlBrowserSessionOptions;
73
+ }
74
+ /**
75
+ * Provisions a dedicated Firecrawl Browser Sandbox session per Mastra thread and connects Playwright over CDP.
76
+ */
77
+ declare class FirecrawlAgentBrowserThreadManager extends AgentBrowserThreadManager {
78
+ private readonly firecrawl;
79
+ private readonly resolveWebSocketUrl;
80
+ private readonly sessionOptions;
81
+ constructor(config: FirecrawlAgentBrowserThreadManagerConfig);
82
+ protected createSession(threadId: string): Promise<FirecrawlAgentBrowserSession>;
83
+ protected doDestroySession(session: FirecrawlAgentBrowserSession): Promise<void>;
84
+ }
85
+
86
+ /**
87
+ * Resolve an HTTP CDP endpoint to a WebSocket URL (same behavior as MastraBrowser.resolveWebSocketUrl).
88
+ */
89
+ declare function resolveCdpWebSocketUrl(url: string, logger?: {
90
+ debug?: (message: string) => void;
91
+ warn?: (message: string) => void;
92
+ }): Promise<string>;
93
+
94
+ export { type FirecrawlAgentBrowserSession, FirecrawlAgentBrowserThreadManager, type FirecrawlAgentBrowserThreadManagerConfig, FirecrawlBrowser, type FirecrawlBrowserConfig, type FirecrawlBrowserSessionOptions, resolveCdpWebSocketUrl };
@@ -0,0 +1,94 @@
1
+ import { AgentBrowserConfig, AgentBrowser, AgentBrowserSession, AgentBrowserThreadManager, AgentBrowserThreadManagerConfig } from '@mastra/agent-browser';
2
+ import { BrowserManager } from 'agent-browser';
3
+ import { Firecrawl } from 'firecrawl';
4
+
5
+ /**
6
+ * Options passed to Firecrawl `POST /v2/browser` (see Firecrawl JS SDK `browser()`).
7
+ *
8
+ * **Note:** {@link AgentBrowserConfig.profile} is the Playwright / agent-browser user-data directory
9
+ * (filesystem path). The optional `profile` below is Firecrawl’s **named sandbox profile** for the
10
+ * hosted browser API only—different meaning, same nested key name as required by the SDK.
11
+ */
12
+ interface FirecrawlBrowserSessionOptions {
13
+ /** Max session wall-clock lifetime (seconds, Firecrawl API). */
14
+ ttl?: number;
15
+ /** Idle timeout before the sandbox recycles the session (seconds, Firecrawl API). */
16
+ activityTtl?: number;
17
+ /** When true, Firecrawl may stream WebView frames for the remote session. */
18
+ streamWebView?: boolean;
19
+ /** Firecrawl named profile (not the same as top-level `AgentBrowserConfig.profile`). */
20
+ profile?: {
21
+ /** Saved profile name in Firecrawl. */
22
+ name: string;
23
+ /** Persist profile changes when the session ends. */
24
+ saveChanges?: boolean;
25
+ };
26
+ /** Optional integration label for Firecrawl analytics / routing. */
27
+ integration?: string;
28
+ /** Optional origin hint for the Firecrawl browser session. */
29
+ origin?: string;
30
+ }
31
+ /** Configuration for {@link FirecrawlBrowser}. */
32
+ type FirecrawlBrowserConfig = AgentBrowserConfig & {
33
+ /** Firecrawl API key (or set `FIRECRAWL_API_KEY` in the environment and omit). */
34
+ apiKey?: string;
35
+ /** Base URL for a self-hosted Firecrawl API. */
36
+ apiUrl?: string;
37
+ /**
38
+ * Firecrawl-only session options (`browser()`). Distinct from {@link AgentBrowserConfig.profile}
39
+ * (local Playwright profile path): see {@link FirecrawlBrowserSessionOptions.profile}.
40
+ */
41
+ firecrawl?: FirecrawlBrowserSessionOptions;
42
+ };
43
+
44
+ /**
45
+ * Mastra browser provider backed by [Firecrawl Browser Sandbox](https://docs.firecrawl.dev/features/browser):
46
+ * provisions remote sessions via API and drives them with the same deterministic tools as {@link AgentBrowser}.
47
+ */
48
+ declare class FirecrawlBrowser extends AgentBrowser {
49
+ readonly name = "FirecrawlBrowser";
50
+ readonly provider = "firecrawl/browser-sandbox";
51
+ /** Narrowed from base `MastraBrowser` (`unknown`) — same pattern as {@link AgentBrowser}. */
52
+ protected sharedManager: BrowserManager | null;
53
+ private readonly firecrawl;
54
+ private readonly sessionOpts;
55
+ private sharedFirecrawlSessionId?;
56
+ constructor(config: FirecrawlBrowserConfig);
57
+ protected doLaunch(): Promise<void>;
58
+ protected doClose(): Promise<void>;
59
+ }
60
+
61
+ /**
62
+ * Thread session with Firecrawl sandbox id for cleanup.
63
+ */
64
+ interface FirecrawlAgentBrowserSession extends AgentBrowserSession {
65
+ firecrawlSessionId?: string;
66
+ }
67
+ interface FirecrawlAgentBrowserThreadManagerConfig extends AgentBrowserThreadManagerConfig {
68
+ firecrawl: Firecrawl;
69
+ /** Resolve HTTP CDP URL to WebSocket URL. */
70
+ resolveWebSocketUrl: (url: string) => Promise<string>;
71
+ /** Options for each `firecrawl.browser()` call (thread scope = one call per thread). */
72
+ sessionOptions?: FirecrawlBrowserSessionOptions;
73
+ }
74
+ /**
75
+ * Provisions a dedicated Firecrawl Browser Sandbox session per Mastra thread and connects Playwright over CDP.
76
+ */
77
+ declare class FirecrawlAgentBrowserThreadManager extends AgentBrowserThreadManager {
78
+ private readonly firecrawl;
79
+ private readonly resolveWebSocketUrl;
80
+ private readonly sessionOptions;
81
+ constructor(config: FirecrawlAgentBrowserThreadManagerConfig);
82
+ protected createSession(threadId: string): Promise<FirecrawlAgentBrowserSession>;
83
+ protected doDestroySession(session: FirecrawlAgentBrowserSession): Promise<void>;
84
+ }
85
+
86
+ /**
87
+ * Resolve an HTTP CDP endpoint to a WebSocket URL (same behavior as MastraBrowser.resolveWebSocketUrl).
88
+ */
89
+ declare function resolveCdpWebSocketUrl(url: string, logger?: {
90
+ debug?: (message: string) => void;
91
+ warn?: (message: string) => void;
92
+ }): Promise<string>;
93
+
94
+ export { type FirecrawlAgentBrowserSession, FirecrawlAgentBrowserThreadManager, type FirecrawlAgentBrowserThreadManagerConfig, FirecrawlBrowser, type FirecrawlBrowserConfig, type FirecrawlBrowserSessionOptions, resolveCdpWebSocketUrl };
package/dist/index.js ADDED
@@ -0,0 +1,255 @@
1
+ import { AgentBrowserThreadManager, AgentBrowser } from '@mastra/agent-browser';
2
+ import { BrowserManager } from 'agent-browser';
3
+ import { Firecrawl } from 'firecrawl';
4
+
5
+ // src/firecrawl-browser.ts
6
+ var FirecrawlAgentBrowserThreadManager = class extends AgentBrowserThreadManager {
7
+ firecrawl;
8
+ resolveWebSocketUrl;
9
+ sessionOptions;
10
+ constructor(config) {
11
+ super(config);
12
+ this.firecrawl = config.firecrawl;
13
+ this.resolveWebSocketUrl = config.resolveWebSocketUrl;
14
+ this.sessionOptions = config.sessionOptions ?? {};
15
+ }
16
+ async createSession(threadId) {
17
+ const savedState = this.getSavedBrowserState(threadId);
18
+ const session = {
19
+ threadId,
20
+ createdAt: Date.now(),
21
+ browserState: savedState
22
+ };
23
+ if (this.scope === "thread") {
24
+ const createRes = await this.firecrawl.browser({
25
+ ttl: this.sessionOptions.ttl,
26
+ activityTtl: this.sessionOptions.activityTtl,
27
+ streamWebView: this.sessionOptions.streamWebView,
28
+ profile: this.sessionOptions.profile,
29
+ integration: this.sessionOptions.integration,
30
+ origin: this.sessionOptions.origin
31
+ });
32
+ if (!createRes.success || !createRes.id || !createRes.cdpUrl) {
33
+ const msg = createRes.error ?? "Firecrawl browser session creation failed";
34
+ const err = new Error(`Firecrawl browser(): ${msg}`);
35
+ if (createRes.id) {
36
+ try {
37
+ await this.firecrawl.deleteBrowser(createRes.id);
38
+ } catch (cleanupErr) {
39
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);
40
+ }
41
+ }
42
+ throw err;
43
+ }
44
+ session.firecrawlSessionId = createRes.id;
45
+ const manager = new BrowserManager();
46
+ const wsUrl = await this.resolveWebSocketUrl(createRes.cdpUrl);
47
+ const launchOptions = {
48
+ headless: this.browserConfig.headless ?? true,
49
+ viewport: this.browserConfig.viewport,
50
+ profile: this.browserConfig.profile,
51
+ executablePath: this.browserConfig.executablePath,
52
+ storageState: this.browserConfig.storageState,
53
+ cdpUrl: wsUrl
54
+ };
55
+ try {
56
+ await manager.launch(launchOptions);
57
+ } catch (error) {
58
+ try {
59
+ await manager.close();
60
+ } catch {
61
+ }
62
+ try {
63
+ await this.firecrawl.deleteBrowser(createRes.id);
64
+ } catch {
65
+ }
66
+ throw error;
67
+ }
68
+ session.manager = manager;
69
+ this.threadManagers.set(threadId, manager);
70
+ try {
71
+ if (savedState && savedState.tabs.length > 0) {
72
+ this.logger?.debug?.(`Restoring browser state for thread ${threadId}: ${savedState.tabs.length} tabs`);
73
+ await this.restoreBrowserState(manager, savedState);
74
+ }
75
+ this.onBrowserCreated?.(manager, threadId);
76
+ } catch (error) {
77
+ this.threadManagers.delete(threadId);
78
+ session.manager = void 0;
79
+ try {
80
+ await manager.close();
81
+ } catch {
82
+ }
83
+ if (session.firecrawlSessionId) {
84
+ try {
85
+ await this.firecrawl.deleteBrowser(session.firecrawlSessionId);
86
+ } catch {
87
+ }
88
+ }
89
+ throw error;
90
+ }
91
+ }
92
+ return session;
93
+ }
94
+ async doDestroySession(session) {
95
+ if (this.scope === "thread" && session.manager) {
96
+ try {
97
+ await session.manager.close();
98
+ } catch {
99
+ }
100
+ this.threadManagers.delete(session.threadId);
101
+ }
102
+ if (session.firecrawlSessionId) {
103
+ try {
104
+ await this.firecrawl.deleteBrowser(session.firecrawlSessionId);
105
+ } catch {
106
+ }
107
+ }
108
+ }
109
+ };
110
+
111
+ // src/resolve-cdp.ts
112
+ async function resolveCdpWebSocketUrl(url, logger) {
113
+ if (url.startsWith("ws://") || url.startsWith("wss://")) {
114
+ return url;
115
+ }
116
+ if (url.startsWith("http://") || url.startsWith("https://")) {
117
+ const baseUrl = url.replace(/\/$/, "");
118
+ const versionUrl = `${baseUrl}/json/version`;
119
+ logger?.debug?.(`Resolving WebSocket URL from ${versionUrl}`);
120
+ const controller = new AbortController();
121
+ const timeoutId = setTimeout(() => controller.abort(), 1e4);
122
+ try {
123
+ const response = await fetch(versionUrl, { signal: controller.signal });
124
+ if (!response.ok) {
125
+ throw new Error(
126
+ `Failed to fetch CDP version info from ${versionUrl}: ${response.status} ${response.statusText}`
127
+ );
128
+ }
129
+ const data = await response.json();
130
+ if (!data.webSocketDebuggerUrl) {
131
+ throw new Error(`No webSocketDebuggerUrl found in CDP version response from ${versionUrl}`);
132
+ }
133
+ logger?.debug?.(`Resolved WebSocket URL: ${data.webSocketDebuggerUrl}`);
134
+ return data.webSocketDebuggerUrl;
135
+ } catch (error) {
136
+ if (error instanceof Error && error.name === "AbortError") {
137
+ throw new Error(`Timeout resolving WebSocket URL from ${versionUrl} (10s)`);
138
+ }
139
+ throw error;
140
+ } finally {
141
+ clearTimeout(timeoutId);
142
+ }
143
+ }
144
+ return url;
145
+ }
146
+
147
+ // src/firecrawl-browser.ts
148
+ function pickSessionOpts(c) {
149
+ return c.firecrawl ?? {};
150
+ }
151
+ function toBaseConfig(config) {
152
+ const { apiKey: _a, apiUrl: _u, firecrawl: _f, ...rest } = config;
153
+ return rest;
154
+ }
155
+ var FirecrawlBrowser = class extends AgentBrowser {
156
+ name = "FirecrawlBrowser";
157
+ provider = "firecrawl/browser-sandbox";
158
+ firecrawl;
159
+ sessionOpts;
160
+ sharedFirecrawlSessionId;
161
+ constructor(config) {
162
+ const apiKey = config.apiKey ?? process.env.FIRECRAWL_API_KEY;
163
+ if (!apiKey) {
164
+ throw new Error("FirecrawlBrowser requires `apiKey` or FIRECRAWL_API_KEY");
165
+ }
166
+ const fc = new Firecrawl({ apiKey, apiUrl: config.apiUrl });
167
+ const sessionOpts = pickSessionOpts(config);
168
+ super({
169
+ ...toBaseConfig(config),
170
+ createThreadManager: (opts) => new FirecrawlAgentBrowserThreadManager({
171
+ ...opts,
172
+ firecrawl: fc,
173
+ resolveWebSocketUrl: (url) => resolveCdpWebSocketUrl(url, opts.logger),
174
+ sessionOptions: sessionOpts
175
+ })
176
+ });
177
+ this.firecrawl = fc;
178
+ this.sessionOpts = sessionOpts;
179
+ }
180
+ async doLaunch() {
181
+ const scope = this.threadManager.getScope();
182
+ if (scope === "thread") {
183
+ await super.doLaunch();
184
+ return;
185
+ }
186
+ const createRes = await this.firecrawl.browser({
187
+ ttl: this.sessionOpts.ttl,
188
+ activityTtl: this.sessionOpts.activityTtl,
189
+ streamWebView: this.sessionOpts.streamWebView,
190
+ profile: this.sessionOpts.profile,
191
+ integration: this.sessionOpts.integration,
192
+ origin: this.sessionOpts.origin
193
+ });
194
+ if (!createRes.success || !createRes.id || !createRes.cdpUrl) {
195
+ const msg = createRes.error ?? "Firecrawl browser session creation failed";
196
+ const err = new Error(`Firecrawl browser(): ${msg}`);
197
+ if (createRes.id) {
198
+ try {
199
+ await this.firecrawl.deleteBrowser(createRes.id);
200
+ } catch (cleanupErr) {
201
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);
202
+ }
203
+ }
204
+ throw err;
205
+ }
206
+ const sessionId = createRes.id;
207
+ this.sharedManager = new BrowserManager();
208
+ try {
209
+ const localConfig = this.config;
210
+ const wsUrl = await resolveCdpWebSocketUrl(createRes.cdpUrl, this.logger);
211
+ const launchOptions = {
212
+ headless: localConfig.headless ?? true,
213
+ viewport: localConfig.viewport,
214
+ profile: localConfig.profile,
215
+ executablePath: localConfig.executablePath,
216
+ storageState: localConfig.storageState,
217
+ cdpUrl: wsUrl
218
+ };
219
+ await this.sharedManager.launch(launchOptions);
220
+ this.threadManager.setSharedManager(this.sharedManager);
221
+ this.setupCloseListenerForSharedScope(this.sharedManager);
222
+ this.sharedFirecrawlSessionId = sessionId;
223
+ } catch (launchErr) {
224
+ try {
225
+ await this.sharedManager.close();
226
+ } catch (closeErr) {
227
+ this.logger?.warn?.(`BrowserManager.close() after failed shared launch: ${closeErr}`);
228
+ }
229
+ try {
230
+ await this.firecrawl.deleteBrowser(sessionId);
231
+ } catch (delErr) {
232
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${sessionId}) after failed shared launch: ${delErr}`);
233
+ }
234
+ this.sharedManager = null;
235
+ this.sharedFirecrawlSessionId = void 0;
236
+ throw launchErr;
237
+ }
238
+ }
239
+ async doClose() {
240
+ const sid = this.sharedFirecrawlSessionId;
241
+ await super.doClose();
242
+ if (sid) {
243
+ try {
244
+ await this.firecrawl.deleteBrowser(sid);
245
+ } catch (err) {
246
+ this.logger?.warn?.(`Firecrawl deleteBrowser(${sid}) failed: ${err}`);
247
+ }
248
+ this.sharedFirecrawlSessionId = void 0;
249
+ }
250
+ }
251
+ };
252
+
253
+ export { FirecrawlAgentBrowserThreadManager, FirecrawlBrowser, resolveCdpWebSocketUrl };
254
+ //# sourceMappingURL=index.js.map
255
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/firecrawl-thread-manager.ts","../src/resolve-cdp.ts","../src/firecrawl-browser.ts"],"names":["BrowserManager"],"mappings":"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","file":"index.js","sourcesContent":["import type { AgentBrowserSession, AgentBrowserThreadManagerConfig } from '@mastra/agent-browser';\nimport { AgentBrowserThreadManager } from '@mastra/agent-browser';\nimport type { BrowserLaunchOptions } from 'agent-browser';\nimport { BrowserManager } from 'agent-browser';\nimport type { Firecrawl } from 'firecrawl';\nimport type { FirecrawlBrowserSessionOptions } from './types';\n\n/**\n * Thread session with Firecrawl sandbox id for cleanup.\n */\nexport interface FirecrawlAgentBrowserSession extends AgentBrowserSession {\n firecrawlSessionId?: string;\n}\n\nexport interface FirecrawlAgentBrowserThreadManagerConfig extends AgentBrowserThreadManagerConfig {\n firecrawl: Firecrawl;\n /** Resolve HTTP CDP URL to WebSocket URL. */\n resolveWebSocketUrl: (url: string) => Promise<string>;\n /** Options for each `firecrawl.browser()` call (thread scope = one call per thread). */\n sessionOptions?: FirecrawlBrowserSessionOptions;\n}\n\n/**\n * Provisions a dedicated Firecrawl Browser Sandbox session per Mastra thread and connects Playwright over CDP.\n */\nexport class FirecrawlAgentBrowserThreadManager extends AgentBrowserThreadManager {\n private readonly firecrawl: Firecrawl;\n private readonly resolveWebSocketUrl: (url: string) => Promise<string>;\n private readonly sessionOptions: FirecrawlBrowserSessionOptions;\n\n constructor(config: FirecrawlAgentBrowserThreadManagerConfig) {\n super(config);\n this.firecrawl = config.firecrawl;\n this.resolveWebSocketUrl = config.resolveWebSocketUrl;\n this.sessionOptions = config.sessionOptions ?? {};\n }\n\n protected override async createSession(threadId: string): Promise<FirecrawlAgentBrowserSession> {\n const savedState = this.getSavedBrowserState(threadId);\n\n const session: FirecrawlAgentBrowserSession = {\n threadId,\n createdAt: Date.now(),\n browserState: savedState,\n };\n\n if (this.scope === 'thread') {\n const createRes = await this.firecrawl.browser({\n ttl: this.sessionOptions.ttl,\n activityTtl: this.sessionOptions.activityTtl,\n streamWebView: this.sessionOptions.streamWebView,\n profile: this.sessionOptions.profile,\n integration: this.sessionOptions.integration,\n origin: this.sessionOptions.origin,\n });\n\n if (!createRes.success || !createRes.id || !createRes.cdpUrl) {\n const msg = createRes.error ?? 'Firecrawl browser session creation failed';\n const err = new Error(`Firecrawl browser(): ${msg}`);\n if (createRes.id) {\n try {\n await this.firecrawl.deleteBrowser(createRes.id);\n } catch (cleanupErr) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);\n }\n }\n throw err;\n }\n\n session.firecrawlSessionId = createRes.id;\n\n const manager = new BrowserManager();\n\n const wsUrl = await this.resolveWebSocketUrl(createRes.cdpUrl);\n\n const launchOptions: BrowserLaunchOptions = {\n headless: this.browserConfig.headless ?? true,\n viewport: this.browserConfig.viewport,\n profile: this.browserConfig.profile,\n executablePath: this.browserConfig.executablePath,\n storageState: this.browserConfig.storageState,\n cdpUrl: wsUrl,\n };\n\n try {\n await manager.launch(launchOptions);\n } catch (error) {\n try {\n await manager.close();\n } catch {\n // ignore\n }\n try {\n await this.firecrawl.deleteBrowser(createRes.id);\n } catch {\n // ignore\n }\n throw error;\n }\n\n session.manager = manager;\n this.threadManagers.set(threadId, manager);\n\n try {\n if (savedState && savedState.tabs.length > 0) {\n this.logger?.debug?.(`Restoring browser state for thread ${threadId}: ${savedState.tabs.length} tabs`);\n await this.restoreBrowserState(manager, savedState);\n }\n this.onBrowserCreated?.(manager, threadId);\n } catch (error) {\n this.threadManagers.delete(threadId);\n session.manager = undefined;\n try {\n await manager.close();\n } catch {\n // ignore\n }\n if (session.firecrawlSessionId) {\n try {\n await this.firecrawl.deleteBrowser(session.firecrawlSessionId);\n } catch {\n // ignore\n }\n }\n throw error;\n }\n }\n\n return session;\n }\n\n protected override async doDestroySession(session: FirecrawlAgentBrowserSession): Promise<void> {\n if (this.scope === 'thread' && session.manager) {\n try {\n await session.manager.close();\n } catch {\n // ignore\n }\n this.threadManagers.delete(session.threadId);\n }\n\n if (session.firecrawlSessionId) {\n try {\n await this.firecrawl.deleteBrowser(session.firecrawlSessionId);\n } catch {\n // ignore\n }\n }\n }\n}\n","/**\n * Resolve an HTTP CDP endpoint to a WebSocket URL (same behavior as MastraBrowser.resolveWebSocketUrl).\n */\nexport async function resolveCdpWebSocketUrl(\n url: string,\n logger?: { debug?: (message: string) => void; warn?: (message: string) => void },\n): Promise<string> {\n if (url.startsWith('ws://') || url.startsWith('wss://')) {\n return url;\n }\n\n if (url.startsWith('http://') || url.startsWith('https://')) {\n const baseUrl = url.replace(/\\/$/, '');\n const versionUrl = `${baseUrl}/json/version`;\n\n logger?.debug?.(`Resolving WebSocket URL from ${versionUrl}`);\n\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), 10000);\n\n try {\n const response = await fetch(versionUrl, { signal: controller.signal });\n\n if (!response.ok) {\n throw new Error(\n `Failed to fetch CDP version info from ${versionUrl}: ${response.status} ${response.statusText}`,\n );\n }\n\n const data = (await response.json()) as { webSocketDebuggerUrl?: string };\n if (!data.webSocketDebuggerUrl) {\n throw new Error(`No webSocketDebuggerUrl found in CDP version response from ${versionUrl}`);\n }\n\n logger?.debug?.(`Resolved WebSocket URL: ${data.webSocketDebuggerUrl}`);\n return data.webSocketDebuggerUrl;\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n throw new Error(`Timeout resolving WebSocket URL from ${versionUrl} (10s)`);\n }\n throw error;\n } finally {\n clearTimeout(timeoutId);\n }\n }\n\n return url;\n}\n","import type { AgentBrowserConfig } from '@mastra/agent-browser';\nimport { AgentBrowser } from '@mastra/agent-browser';\nimport type { BrowserLaunchOptions } from 'agent-browser';\nimport { BrowserManager } from 'agent-browser';\nimport { Firecrawl } from 'firecrawl';\nimport { FirecrawlAgentBrowserThreadManager } from './firecrawl-thread-manager';\nimport { resolveCdpWebSocketUrl } from './resolve-cdp';\nimport type { FirecrawlBrowserConfig, FirecrawlBrowserSessionOptions } from './types';\n\nfunction pickSessionOpts(c: FirecrawlBrowserConfig): FirecrawlBrowserSessionOptions {\n return c.firecrawl ?? {};\n}\n\nfunction toBaseConfig(config: FirecrawlBrowserConfig): AgentBrowserConfig {\n const { apiKey: _a, apiUrl: _u, firecrawl: _f, ...rest } = config;\n return rest;\n}\n\n/**\n * Mastra browser provider backed by [Firecrawl Browser Sandbox](https://docs.firecrawl.dev/features/browser):\n * provisions remote sessions via API and drives them with the same deterministic tools as {@link AgentBrowser}.\n */\nexport class FirecrawlBrowser extends AgentBrowser {\n override readonly name = 'FirecrawlBrowser';\n override readonly provider = 'firecrawl/browser-sandbox';\n\n /** Narrowed from base `MastraBrowser` (`unknown`) — same pattern as {@link AgentBrowser}. */\n declare protected sharedManager: BrowserManager | null;\n\n private readonly firecrawl: Firecrawl;\n private readonly sessionOpts: FirecrawlBrowserSessionOptions;\n private sharedFirecrawlSessionId?: string;\n\n constructor(config: FirecrawlBrowserConfig) {\n const apiKey = config.apiKey ?? process.env.FIRECRAWL_API_KEY;\n if (!apiKey) {\n throw new Error('FirecrawlBrowser requires `apiKey` or FIRECRAWL_API_KEY');\n }\n const fc = new Firecrawl({ apiKey, apiUrl: config.apiUrl });\n const sessionOpts = pickSessionOpts(config);\n\n super({\n ...toBaseConfig(config),\n createThreadManager: opts =>\n new FirecrawlAgentBrowserThreadManager({\n ...opts,\n firecrawl: fc,\n resolveWebSocketUrl: url => resolveCdpWebSocketUrl(url, opts.logger),\n sessionOptions: sessionOpts,\n }),\n });\n this.firecrawl = fc;\n this.sessionOpts = sessionOpts;\n }\n\n protected override async doLaunch(): Promise<void> {\n const scope = this.threadManager.getScope();\n if (scope === 'thread') {\n await super.doLaunch();\n return;\n }\n\n const createRes = await this.firecrawl.browser({\n ttl: this.sessionOpts.ttl,\n activityTtl: this.sessionOpts.activityTtl,\n streamWebView: this.sessionOpts.streamWebView,\n profile: this.sessionOpts.profile,\n integration: this.sessionOpts.integration,\n origin: this.sessionOpts.origin,\n });\n\n if (!createRes.success || !createRes.id || !createRes.cdpUrl) {\n const msg = createRes.error ?? 'Firecrawl browser session creation failed';\n const err = new Error(`Firecrawl browser(): ${msg}`);\n if (createRes.id) {\n try {\n await this.firecrawl.deleteBrowser(createRes.id);\n } catch (cleanupErr) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${createRes.id}) after failed browser(): ${cleanupErr}`);\n }\n }\n throw err;\n }\n\n const sessionId = createRes.id;\n this.sharedManager = new BrowserManager();\n\n try {\n const localConfig = this.config as AgentBrowserConfig;\n const wsUrl = await resolveCdpWebSocketUrl(createRes.cdpUrl, this.logger);\n\n const launchOptions: BrowserLaunchOptions = {\n headless: localConfig.headless ?? true,\n viewport: localConfig.viewport,\n profile: localConfig.profile,\n executablePath: localConfig.executablePath,\n storageState: localConfig.storageState,\n cdpUrl: wsUrl,\n };\n\n await this.sharedManager.launch(launchOptions);\n this.threadManager.setSharedManager(this.sharedManager);\n this.setupCloseListenerForSharedScope(this.sharedManager);\n this.sharedFirecrawlSessionId = sessionId;\n } catch (launchErr) {\n try {\n await this.sharedManager.close();\n } catch (closeErr) {\n this.logger?.warn?.(`BrowserManager.close() after failed shared launch: ${closeErr}`);\n }\n try {\n await this.firecrawl.deleteBrowser(sessionId);\n } catch (delErr) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${sessionId}) after failed shared launch: ${delErr}`);\n }\n this.sharedManager = null;\n this.sharedFirecrawlSessionId = undefined;\n throw launchErr;\n }\n }\n\n protected override async doClose(): Promise<void> {\n const sid = this.sharedFirecrawlSessionId;\n await super.doClose();\n if (sid) {\n try {\n await this.firecrawl.deleteBrowser(sid);\n } catch (err) {\n this.logger?.warn?.(`Firecrawl deleteBrowser(${sid}) failed: ${err}`);\n }\n this.sharedFirecrawlSessionId = undefined;\n }\n }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@mastra/browser-firecrawl",
3
- "version": "0.0.0",
3
+ "version": "0.1.0-alpha.0",
4
4
  "description": "Mastra browser automation using Firecrawl Browser Sandbox (hosted CDP + agent-browser tools)",
5
5
  "type": "module",
6
6
  "files": [
@@ -22,30 +22,24 @@
22
22
  },
23
23
  "./package.json": "./package.json"
24
24
  },
25
- "scripts": {
26
- "build": "tsup --silent --config tsup.config.ts",
27
- "build:watch": "tsup --watch --silent --config tsup.config.ts",
28
- "test": "vitest run",
29
- "lint": "eslint ."
30
- },
31
25
  "license": "Apache-2.0",
32
26
  "dependencies": {
33
27
  "firecrawl": "^4.18.4"
34
28
  },
35
29
  "devDependencies": {
36
- "@internal/lint": "workspace:*",
37
- "@internal/types-builder": "workspace:*",
38
- "@mastra/agent-browser": "workspace:*",
39
- "@mastra/core": "workspace:*",
40
30
  "@types/node": "22.19.15",
41
- "@vitest/coverage-v8": "catalog:",
42
- "@vitest/ui": "catalog:",
31
+ "@vitest/coverage-v8": "4.1.5",
32
+ "@vitest/ui": "4.1.5",
43
33
  "agent-browser": "0.19.0",
44
34
  "eslint": "^10.2.1",
45
35
  "playwright-core": "^1.57.0",
46
36
  "tsup": "^8.5.1",
47
- "typescript": "catalog:",
48
- "vitest": "catalog:"
37
+ "typescript": "^6.0.3",
38
+ "vitest": "4.1.5",
39
+ "@internal/types-builder": "0.0.74",
40
+ "@mastra/agent-browser": "0.3.0-alpha.0",
41
+ "@internal/lint": "0.0.99",
42
+ "@mastra/core": "1.38.0-alpha.3"
49
43
  },
50
44
  "peerDependencies": {
51
45
  "@mastra/agent-browser": ">=0.3.0 <1.0.0",
@@ -63,5 +57,11 @@
63
57
  },
64
58
  "engines": {
65
59
  "node": ">=22.13.0"
60
+ },
61
+ "scripts": {
62
+ "build": "tsup --silent --config tsup.config.ts",
63
+ "build:watch": "tsup --watch --silent --config tsup.config.ts",
64
+ "test": "vitest run",
65
+ "lint": "eslint ."
66
66
  }
67
- }
67
+ }