ai-browser 0.2.0 → 0.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (98) hide show
  1. package/README.md +76 -7
  2. package/dist/agent/agent-loop.d.ts.map +1 -1
  3. package/dist/agent/agent-loop.js +2 -1
  4. package/dist/agent/agent-loop.js.map +1 -1
  5. package/dist/api/mcp-sse.d.ts.map +1 -1
  6. package/dist/api/mcp-sse.js +26 -8
  7. package/dist/api/mcp-sse.js.map +1 -1
  8. package/dist/api/routes.js +1 -1
  9. package/dist/api/routes.js.map +1 -1
  10. package/dist/browser/BrowserManager.d.ts +2 -1
  11. package/dist/browser/BrowserManager.d.ts.map +1 -1
  12. package/dist/browser/BrowserManager.js +50 -7
  13. package/dist/browser/BrowserManager.js.map +1 -1
  14. package/dist/browser/PageEventTracker.d.ts +1 -1
  15. package/dist/browser/PageEventTracker.d.ts.map +1 -1
  16. package/dist/browser/SessionManager.d.ts +1 -1
  17. package/dist/browser/SessionManager.d.ts.map +1 -1
  18. package/dist/browser/actions.d.ts +1 -1
  19. package/dist/browser/actions.d.ts.map +1 -1
  20. package/dist/browser/index.d.ts +1 -1
  21. package/dist/browser/index.d.ts.map +1 -1
  22. package/dist/cli/mcp-stdio.js +1 -1
  23. package/dist/cli/mcp-stdio.js.map +1 -1
  24. package/dist/mcp/browser-mcp-server.d.ts +6 -9
  25. package/dist/mcp/browser-mcp-server.d.ts.map +1 -1
  26. package/dist/mcp/browser-mcp-server.js +125 -218
  27. package/dist/mcp/browser-mcp-server.js.map +1 -1
  28. package/dist/mcp/task-tools.d.ts +6 -0
  29. package/dist/mcp/task-tools.d.ts.map +1 -0
  30. package/dist/mcp/task-tools.js +303 -0
  31. package/dist/mcp/task-tools.js.map +1 -0
  32. package/dist/semantic/ContentExtractor.d.ts +1 -1
  33. package/dist/semantic/ContentExtractor.d.ts.map +1 -1
  34. package/dist/semantic/ElementCollector.d.ts +1 -1
  35. package/dist/semantic/ElementCollector.d.ts.map +1 -1
  36. package/dist/semantic/IframeHandler.d.ts +1 -1
  37. package/dist/semantic/IframeHandler.d.ts.map +1 -1
  38. package/dist/semantic/PageAnalyzer.d.ts +1 -1
  39. package/dist/semantic/PageAnalyzer.d.ts.map +1 -1
  40. package/dist/semantic/RegionDetector.d.ts +1 -1
  41. package/dist/semantic/RegionDetector.d.ts.map +1 -1
  42. package/dist/semantic/StateTracker.d.ts +1 -1
  43. package/dist/semantic/StateTracker.d.ts.map +1 -1
  44. package/dist/task/artifact-store.d.ts +36 -0
  45. package/dist/task/artifact-store.d.ts.map +1 -0
  46. package/dist/task/artifact-store.js +115 -0
  47. package/dist/task/artifact-store.js.map +1 -0
  48. package/dist/task/cancel-token.d.ts +13 -0
  49. package/dist/task/cancel-token.d.ts.map +1 -0
  50. package/dist/task/cancel-token.js +42 -0
  51. package/dist/task/cancel-token.js.map +1 -0
  52. package/dist/task/error-codes.d.ts +19 -0
  53. package/dist/task/error-codes.d.ts.map +1 -0
  54. package/dist/task/error-codes.js +22 -0
  55. package/dist/task/error-codes.js.map +1 -0
  56. package/dist/task/index.d.ts +17 -0
  57. package/dist/task/index.d.ts.map +1 -0
  58. package/dist/task/index.js +10 -0
  59. package/dist/task/index.js.map +1 -0
  60. package/dist/task/run-manager.d.ts +77 -0
  61. package/dist/task/run-manager.d.ts.map +1 -0
  62. package/dist/task/run-manager.js +286 -0
  63. package/dist/task/run-manager.js.map +1 -0
  64. package/dist/task/run-store.d.ts +39 -0
  65. package/dist/task/run-store.d.ts.map +1 -0
  66. package/dist/task/run-store.js +88 -0
  67. package/dist/task/run-store.js.map +1 -0
  68. package/dist/task/templates/batch-extract.d.ts +33 -0
  69. package/dist/task/templates/batch-extract.d.ts.map +1 -0
  70. package/dist/task/templates/batch-extract.js +153 -0
  71. package/dist/task/templates/batch-extract.js.map +1 -0
  72. package/dist/task/templates/login-keep-session.d.ts +34 -0
  73. package/dist/task/templates/login-keep-session.d.ts.map +1 -0
  74. package/dist/task/templates/login-keep-session.js +190 -0
  75. package/dist/task/templates/login-keep-session.js.map +1 -0
  76. package/dist/task/templates/multi-tab-compare.d.ts +43 -0
  77. package/dist/task/templates/multi-tab-compare.d.ts.map +1 -0
  78. package/dist/task/templates/multi-tab-compare.js +204 -0
  79. package/dist/task/templates/multi-tab-compare.js.map +1 -0
  80. package/dist/task/templates/registry.d.ts +13 -0
  81. package/dist/task/templates/registry.d.ts.map +1 -0
  82. package/dist/task/templates/registry.js +40 -0
  83. package/dist/task/templates/registry.js.map +1 -0
  84. package/dist/task/tool-actions.d.ts +114 -0
  85. package/dist/task/tool-actions.d.ts.map +1 -0
  86. package/dist/task/tool-actions.js +371 -0
  87. package/dist/task/tool-actions.js.map +1 -0
  88. package/dist/task/tool-context.d.ts +26 -0
  89. package/dist/task/tool-context.d.ts.map +1 -0
  90. package/dist/task/tool-context.js +2 -0
  91. package/dist/task/tool-context.js.map +1 -0
  92. package/dist/utils/url-validator.d.ts +13 -0
  93. package/dist/utils/url-validator.d.ts.map +1 -1
  94. package/dist/utils/url-validator.js +64 -0
  95. package/dist/utils/url-validator.js.map +1 -1
  96. package/package.json +4 -3
  97. package/public/agent.html +3 -927
  98. package/public/index.html +1910 -664
@@ -0,0 +1,286 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import { CancelToken } from './cancel-token.js';
3
+ import { ErrorCode } from './error-codes.js';
4
+ const TERMINAL_STATUSES = new Set([
5
+ 'succeeded', 'failed', 'partial_success', 'canceled',
6
+ ]);
7
+ // ===== Semaphore for concurrency control =====
8
+ class Semaphore {
9
+ _count;
10
+ _waiters = [];
11
+ constructor(max) {
12
+ this._count = max;
13
+ }
14
+ async acquire() {
15
+ if (this._count > 0) {
16
+ this._count--;
17
+ return;
18
+ }
19
+ return new Promise((resolve) => {
20
+ this._waiters.push(resolve);
21
+ });
22
+ }
23
+ release() {
24
+ const next = this._waiters.shift();
25
+ if (next) {
26
+ next();
27
+ }
28
+ else {
29
+ this._count++;
30
+ }
31
+ }
32
+ }
33
+ // ===== RunManager =====
34
+ const DEFAULT_TTL_MS = 30 * 60 * 1000; // 30 minutes
35
+ const CLEANUP_INTERVAL_MS = 5 * 60 * 1000; // 5 minutes
36
+ const DEFAULT_MAX_CONCURRENT = 5;
37
+ export class RunManager {
38
+ runs = new Map();
39
+ tokens = new Map();
40
+ timers = new Map();
41
+ terminalHooks = new Map();
42
+ cleanupTimer = null;
43
+ ttlMs;
44
+ semaphore;
45
+ constructor(opts) {
46
+ this.ttlMs = opts?.ttlMs ?? DEFAULT_TTL_MS;
47
+ this.semaphore = new Semaphore(opts?.maxConcurrentRuns ?? DEFAULT_MAX_CONCURRENT);
48
+ this.cleanupTimer = setInterval(() => this.cleanup(), CLEANUP_INTERVAL_MS);
49
+ if (this.cleanupTimer.unref)
50
+ this.cleanupTimer.unref();
51
+ }
52
+ /**
53
+ * Submit a run for execution.
54
+ * In sync mode, awaits the executor and returns the result inline.
55
+ * In async mode, fires the executor and returns immediately with runId.
56
+ */
57
+ async submit(templateId, sessionId, ownsSession, totalSteps, executor, opts) {
58
+ const run = this.createRun(templateId, sessionId, ownsSession, totalSteps);
59
+ const token = new CancelToken();
60
+ this.tokens.set(run.runId, token);
61
+ if (opts.onTerminal) {
62
+ this.terminalHooks.set(run.runId, opts.onTerminal);
63
+ }
64
+ if (opts.mode === 'sync') {
65
+ const result = await this.executeRun(run.runId, executor, token, opts.timeoutMs);
66
+ return { runId: run.runId, syncResult: result };
67
+ }
68
+ // Async: fire-and-forget
69
+ this.executeRun(run.runId, executor, token, opts.timeoutMs).catch(() => { });
70
+ return { runId: run.runId };
71
+ }
72
+ get(runId) {
73
+ return this.runs.get(runId);
74
+ }
75
+ cancel(runId) {
76
+ const run = this.runs.get(runId);
77
+ if (!run)
78
+ return false;
79
+ if (TERMINAL_STATUSES.has(run.status))
80
+ return false;
81
+ const token = this.tokens.get(runId);
82
+ if (token)
83
+ token.cancel();
84
+ this.transition(runId, 'canceled');
85
+ run.error = { errorCode: ErrorCode.RUN_CANCELED, message: 'Run canceled by user' };
86
+ return true;
87
+ }
88
+ list(filter) {
89
+ const results = this.filterRuns(filter);
90
+ const offset = filter?.offset ?? 0;
91
+ const limit = filter?.limit ?? 50;
92
+ return results.slice(offset, offset + limit);
93
+ }
94
+ count(filter) {
95
+ return this.filterRuns(filter).length;
96
+ }
97
+ attachArtifact(runId, artifactId) {
98
+ const run = this.runs.get(runId);
99
+ if (!run)
100
+ return;
101
+ run.artifactIds.push(artifactId);
102
+ run.updatedAt = Date.now();
103
+ }
104
+ dispose() {
105
+ if (this.cleanupTimer) {
106
+ clearInterval(this.cleanupTimer);
107
+ this.cleanupTimer = null;
108
+ }
109
+ for (const timer of this.timers.values()) {
110
+ clearTimeout(timer);
111
+ }
112
+ this.timers.clear();
113
+ for (const token of this.tokens.values()) {
114
+ token.cancel();
115
+ }
116
+ this.tokens.clear();
117
+ this.terminalHooks.clear();
118
+ this.runs.clear();
119
+ }
120
+ // --- Internal helpers ---
121
+ createRun(templateId, sessionId, ownsSession, totalSteps) {
122
+ const now = Date.now();
123
+ const run = {
124
+ runId: randomUUID(),
125
+ templateId,
126
+ status: 'queued',
127
+ createdAt: now,
128
+ updatedAt: now,
129
+ sessionId,
130
+ ownsSession,
131
+ progress: { totalSteps, doneSteps: 0 },
132
+ metrics: { elapsedMs: 0 },
133
+ artifactIds: [],
134
+ };
135
+ this.runs.set(run.runId, run);
136
+ return run;
137
+ }
138
+ transition(runId, status) {
139
+ const run = this.runs.get(runId);
140
+ if (!run)
141
+ return;
142
+ if (TERMINAL_STATUSES.has(run.status))
143
+ return;
144
+ run.status = status;
145
+ run.updatedAt = Date.now();
146
+ if (TERMINAL_STATUSES.has(status)) {
147
+ run.metrics.elapsedMs = run.updatedAt - run.createdAt;
148
+ // Cleanup token and timer
149
+ this.tokens.delete(runId);
150
+ const timer = this.timers.get(runId);
151
+ if (timer) {
152
+ clearTimeout(timer);
153
+ this.timers.delete(runId);
154
+ }
155
+ }
156
+ }
157
+ updateProgress(runId, doneSteps) {
158
+ const run = this.runs.get(runId);
159
+ if (!run || TERMINAL_STATUSES.has(run.status))
160
+ return;
161
+ run.progress.doneSteps = doneSteps;
162
+ run.updatedAt = Date.now();
163
+ }
164
+ async executeRun(runId, executor, token, timeoutMs) {
165
+ // Acquire semaphore slot
166
+ await this.semaphore.acquire();
167
+ try {
168
+ // Check if canceled while waiting for semaphore
169
+ if (token.canceled) {
170
+ this.transition(runId, 'canceled');
171
+ return undefined;
172
+ }
173
+ this.transition(runId, 'running');
174
+ // Setup timeout
175
+ const timeout = Math.min(timeoutMs ?? 300_000, 600_000);
176
+ const timeoutTimer = setTimeout(() => {
177
+ const run = this.runs.get(runId);
178
+ if (run && run.status === 'running') {
179
+ token.cancel();
180
+ run.error = {
181
+ errorCode: ErrorCode.RUN_TIMEOUT,
182
+ message: 'Execution timed out',
183
+ };
184
+ this.transition(runId, 'failed');
185
+ }
186
+ }, timeout);
187
+ this.timers.set(runId, timeoutTimer);
188
+ // Execute
189
+ const result = await executor(runId, token, (done) => this.updateProgress(runId, done));
190
+ // Check if already transitioned (timeout/cancel)
191
+ const run = this.runs.get(runId);
192
+ if (!run) {
193
+ return undefined;
194
+ }
195
+ if (TERMINAL_STATUSES.has(run.status)) {
196
+ // Preserve partial results even if status has already been finalized.
197
+ if (run.result === undefined && result !== undefined) {
198
+ run.result = result;
199
+ }
200
+ return run.result;
201
+ }
202
+ // Determine final status from result
203
+ run.result = result;
204
+ const status = this.determineStatus(result);
205
+ this.transition(runId, status);
206
+ return result;
207
+ }
208
+ catch (err) {
209
+ const run = this.runs.get(runId);
210
+ if (run && TERMINAL_STATUSES.has(run.status)) {
211
+ return run.result;
212
+ }
213
+ if (run && !TERMINAL_STATUSES.has(run.status)) {
214
+ run.error = {
215
+ errorCode: err.errorCode ?? ErrorCode.EXECUTION_ERROR,
216
+ message: err.message || 'Execution failed',
217
+ };
218
+ this.transition(runId, 'failed');
219
+ }
220
+ throw err;
221
+ }
222
+ finally {
223
+ this.semaphore.release();
224
+ await this.fireTerminalHook(runId);
225
+ }
226
+ }
227
+ async fireTerminalHook(runId) {
228
+ const run = this.runs.get(runId);
229
+ if (!run || !TERMINAL_STATUSES.has(run.status)) {
230
+ return;
231
+ }
232
+ const hook = this.terminalHooks.get(runId);
233
+ if (!hook) {
234
+ return;
235
+ }
236
+ this.terminalHooks.delete(runId);
237
+ try {
238
+ await hook(run);
239
+ }
240
+ catch {
241
+ // Ignore terminal hook failures to avoid masking run status.
242
+ }
243
+ }
244
+ determineStatus(result) {
245
+ if (result?.summary && typeof result.summary.succeeded === 'number' && typeof result.summary.total === 'number') {
246
+ const { succeeded, total } = result.summary;
247
+ if (succeeded === total)
248
+ return 'succeeded';
249
+ if (total > 0 && succeeded / total >= 0.5)
250
+ return 'partial_success';
251
+ return 'failed';
252
+ }
253
+ if (typeof result?.success === 'boolean') {
254
+ return result.success ? 'succeeded' : 'failed';
255
+ }
256
+ return 'succeeded';
257
+ }
258
+ filterRuns(filter) {
259
+ let results = Array.from(this.runs.values());
260
+ if (filter?.status) {
261
+ results = results.filter((r) => r.status === filter.status);
262
+ }
263
+ if (filter?.templateId) {
264
+ results = results.filter((r) => r.templateId === filter.templateId);
265
+ }
266
+ // Sort by createdAt descending (newest first)
267
+ results.sort((a, b) => b.createdAt - a.createdAt);
268
+ return results;
269
+ }
270
+ cleanup() {
271
+ const now = Date.now();
272
+ for (const [id, run] of this.runs) {
273
+ if (TERMINAL_STATUSES.has(run.status) && now - run.updatedAt > this.ttlMs) {
274
+ this.runs.delete(id);
275
+ this.tokens.delete(id);
276
+ this.terminalHooks.delete(id);
277
+ const timer = this.timers.get(id);
278
+ if (timer) {
279
+ clearTimeout(timer);
280
+ this.timers.delete(id);
281
+ }
282
+ }
283
+ }
284
+ }
285
+ }
286
+ //# sourceMappingURL=run-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-manager.js","sourceRoot":"","sources":["../../src/task/run-manager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,39 @@
1
+ export type RunStatus = 'queued' | 'running' | 'succeeded' | 'failed' | 'partial_success';
2
+ export interface RunState {
3
+ runId: string;
4
+ templateId: string;
5
+ status: RunStatus;
6
+ createdAt: number;
7
+ updatedAt: number;
8
+ sessionId: string;
9
+ ownsSession: boolean;
10
+ progress: {
11
+ totalSteps: number;
12
+ doneSteps: number;
13
+ };
14
+ metrics: {
15
+ elapsedMs: number;
16
+ };
17
+ result?: any;
18
+ error?: {
19
+ errorCode: string;
20
+ message: string;
21
+ details?: any;
22
+ };
23
+ }
24
+ export declare class RunStore {
25
+ private runs;
26
+ private cleanupTimer;
27
+ private ttlMs;
28
+ constructor(ttlMs?: number);
29
+ create(templateId: string, sessionId: string, ownsSession: boolean, totalSteps: number): RunState;
30
+ get(runId: string): RunState | undefined;
31
+ transition(runId: string, status: RunStatus): void;
32
+ updateProgress(runId: string, doneSteps: number): void;
33
+ setResult(runId: string, result: any): void;
34
+ setError(runId: string, error: RunState['error']): void;
35
+ list(): RunState[];
36
+ dispose(): void;
37
+ private cleanup;
38
+ }
39
+ //# sourceMappingURL=run-store.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-store.d.ts","sourceRoot":"","sources":["../../src/task/run-store.ts"],"names":[],"mappings":"AAIA,MAAM,MAAM,SAAS,GAAG,QAAQ,GAAG,SAAS,GAAG,WAAW,GAAG,QAAQ,GAAG,iBAAiB,CAAC;AAI1F,MAAM,WAAW,QAAQ;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,CAAC;IACnB,MAAM,EAAE,SAAS,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,OAAO,CAAC;IACrB,QAAQ,EAAE;QAAE,UAAU,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC;IACpD,OAAO,EAAE;QAAE,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC;IAC/B,MAAM,CAAC,EAAE,GAAG,CAAC;IACb,KAAK,CAAC,EAAE;QAAE,SAAS,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAC;QAAC,OAAO,CAAC,EAAE,GAAG,CAAA;KAAE,CAAC;CAC/D;AAOD,qBAAa,QAAQ;IACnB,OAAO,CAAC,IAAI,CAA+B;IAC3C,OAAO,CAAC,YAAY,CAA+C;IACnE,OAAO,CAAC,KAAK,CAAS;gBAEV,KAAK,SAAiB;IAOlC,MAAM,CACJ,UAAU,EAAE,MAAM,EAClB,SAAS,EAAE,MAAM,EACjB,WAAW,EAAE,OAAO,EACpB,UAAU,EAAE,MAAM,GACjB,QAAQ;IAiBX,GAAG,CAAC,KAAK,EAAE,MAAM,GAAG,QAAQ,GAAG,SAAS;IAIxC,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,GAAG,IAAI;IAWlD,cAAc,CAAC,KAAK,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,IAAI;IAOtD,SAAS,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,GAAG,IAAI;IAO3C,QAAQ,CAAC,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,IAAI;IAOvD,IAAI,IAAI,QAAQ,EAAE;IAIlB,OAAO,IAAI,IAAI;IAQf,OAAO,CAAC,OAAO;CAQhB"}
@@ -0,0 +1,88 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ const TERMINAL_STATUSES = new Set(['succeeded', 'failed', 'partial_success']);
3
+ // ===== RunStore =====
4
+ const DEFAULT_TTL_MS = 30 * 60 * 1000; // 30 minutes
5
+ const CLEANUP_INTERVAL_MS = 5 * 60 * 1000; // 5 minutes
6
+ export class RunStore {
7
+ runs = new Map();
8
+ cleanupTimer = null;
9
+ ttlMs;
10
+ constructor(ttlMs = DEFAULT_TTL_MS) {
11
+ this.ttlMs = ttlMs;
12
+ this.cleanupTimer = setInterval(() => this.cleanup(), CLEANUP_INTERVAL_MS);
13
+ // Allow Node to exit even if timer is active
14
+ if (this.cleanupTimer.unref)
15
+ this.cleanupTimer.unref();
16
+ }
17
+ create(templateId, sessionId, ownsSession, totalSteps) {
18
+ const now = Date.now();
19
+ const run = {
20
+ runId: randomUUID(),
21
+ templateId,
22
+ status: 'queued',
23
+ createdAt: now,
24
+ updatedAt: now,
25
+ sessionId,
26
+ ownsSession,
27
+ progress: { totalSteps, doneSteps: 0 },
28
+ metrics: { elapsedMs: 0 },
29
+ };
30
+ this.runs.set(run.runId, run);
31
+ return run;
32
+ }
33
+ get(runId) {
34
+ return this.runs.get(runId);
35
+ }
36
+ transition(runId, status) {
37
+ const run = this.runs.get(runId);
38
+ if (!run)
39
+ return;
40
+ if (TERMINAL_STATUSES.has(run.status))
41
+ return; // terminal states are irreversible
42
+ run.status = status;
43
+ run.updatedAt = Date.now();
44
+ if (TERMINAL_STATUSES.has(status)) {
45
+ run.metrics.elapsedMs = run.updatedAt - run.createdAt;
46
+ }
47
+ }
48
+ updateProgress(runId, doneSteps) {
49
+ const run = this.runs.get(runId);
50
+ if (!run)
51
+ return;
52
+ run.progress.doneSteps = doneSteps;
53
+ run.updatedAt = Date.now();
54
+ }
55
+ setResult(runId, result) {
56
+ const run = this.runs.get(runId);
57
+ if (!run)
58
+ return;
59
+ run.result = result;
60
+ run.updatedAt = Date.now();
61
+ }
62
+ setError(runId, error) {
63
+ const run = this.runs.get(runId);
64
+ if (!run)
65
+ return;
66
+ run.error = error;
67
+ run.updatedAt = Date.now();
68
+ }
69
+ list() {
70
+ return Array.from(this.runs.values());
71
+ }
72
+ dispose() {
73
+ if (this.cleanupTimer) {
74
+ clearInterval(this.cleanupTimer);
75
+ this.cleanupTimer = null;
76
+ }
77
+ this.runs.clear();
78
+ }
79
+ cleanup() {
80
+ const now = Date.now();
81
+ for (const [id, run] of this.runs) {
82
+ if (TERMINAL_STATUSES.has(run.status) && now - run.updatedAt > this.ttlMs) {
83
+ this.runs.delete(id);
84
+ }
85
+ }
86
+ }
87
+ }
88
+ //# sourceMappingURL=run-store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-store.js","sourceRoot":"","sources":["../../src/task/run-store.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,33 @@
1
+ import type { ToolContext } from '../tool-context.js';
2
+ import type { CancelToken } from '../cancel-token.js';
3
+ export interface BatchExtractInputs {
4
+ urls: string[];
5
+ extract?: {
6
+ pageInfo?: boolean;
7
+ content?: boolean;
8
+ maxElements?: number;
9
+ maxContentLength?: number;
10
+ };
11
+ concurrency?: number;
12
+ }
13
+ export interface BatchExtractItem {
14
+ url: string;
15
+ title?: string;
16
+ pageType?: string;
17
+ elementCount?: number;
18
+ contentSections?: number;
19
+ content?: any;
20
+ pageInfo?: any;
21
+ success: boolean;
22
+ error?: string;
23
+ }
24
+ export interface BatchExtractResult {
25
+ summary: {
26
+ total: number;
27
+ succeeded: number;
28
+ failed: number;
29
+ };
30
+ items: BatchExtractItem[];
31
+ }
32
+ export declare function executeBatchExtract(ctx: ToolContext, sessionId: string, inputs: BatchExtractInputs, onProgress?: (done: number) => void, token?: CancelToken): Promise<BatchExtractResult>;
33
+ //# sourceMappingURL=batch-extract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"batch-extract.d.ts","sourceRoot":"","sources":["../../../src/task/templates/batch-extract.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACtD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AAMtD,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,OAAO,CAAC,EAAE;QACR,QAAQ,CAAC,EAAE,OAAO,CAAC;QACnB,OAAO,CAAC,EAAE,OAAO,CAAC;QAClB,WAAW,CAAC,EAAE,MAAM,CAAC;QACrB,gBAAgB,CAAC,EAAE,MAAM,CAAC;KAC3B,CAAC;IACF,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,MAAM,WAAW,gBAAgB;IAC/B,GAAG,EAAE,MAAM,CAAC;IACZ,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,OAAO,CAAC,EAAE,GAAG,CAAC;IACd,QAAQ,CAAC,EAAE,GAAG,CAAC;IACf,OAAO,EAAE,OAAO,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,kBAAkB;IACjC,OAAO,EAAE;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,CAAC;IAC9D,KAAK,EAAE,gBAAgB,EAAE,CAAC;CAC3B;AA4HD,wBAAsB,mBAAmB,CACvC,GAAG,EAAE,WAAW,EAChB,SAAS,EAAE,MAAM,EACjB,MAAM,EAAE,kBAAkB,EAC1B,UAAU,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,EACnC,KAAK,CAAC,EAAE,WAAW,GAClB,OAAO,CAAC,kBAAkB,CAAC,CA4E7B"}
@@ -0,0 +1,153 @@
1
+ import * as actions from '../tool-actions.js';
2
+ import { validateUrlAsync } from '../../utils/url-validator.js';
3
+ // Max concurrency cap
4
+ const MAX_CONCURRENCY = 5;
5
+ // Errors worth retrying
6
+ const RETRYABLE_ERRORS = ['NAVIGATION_TIMEOUT', 'PAGE_CRASHED'];
7
+ // ===== Sliding window concurrency =====
8
+ async function slidingWindow(items, concurrency, fn) {
9
+ let nextIndex = 0;
10
+ const workers = [];
11
+ async function worker() {
12
+ while (nextIndex < items.length) {
13
+ const i = nextIndex++;
14
+ await fn(items[i], i);
15
+ }
16
+ }
17
+ for (let i = 0; i < Math.min(concurrency, items.length); i++) {
18
+ workers.push(worker());
19
+ }
20
+ await Promise.all(workers);
21
+ }
22
+ async function extractSingleUrl(ctx, sessionId, url, opts, token) {
23
+ let lastError;
24
+ for (let attempt = 0; attempt < 2; attempt++) {
25
+ if (token?.canceled) {
26
+ return { url, success: false, error: 'Canceled' };
27
+ }
28
+ let tabId;
29
+ try {
30
+ const tab = await actions.createTab(ctx, sessionId, url);
31
+ tabId = tab.tabId;
32
+ if (token?.canceled) {
33
+ await actions.closeTab(ctx, sessionId, tabId).catch(() => { });
34
+ return { url, success: false, error: 'Canceled' };
35
+ }
36
+ // Wait for DOM stability
37
+ await actions.waitForStable(ctx, sessionId, tabId, { timeout: 10000 });
38
+ const item = { url, success: true };
39
+ // Extract page info
40
+ if (opts.pageInfo) {
41
+ const info = await actions.getPageInfo(ctx, sessionId, tabId, {
42
+ maxElements: opts.maxElements,
43
+ visibleOnly: false,
44
+ });
45
+ item.title = info.page.title;
46
+ item.pageType = info.page.type;
47
+ item.elementCount = info.totalElements;
48
+ item.pageInfo = info;
49
+ }
50
+ // Extract page content
51
+ if (opts.content) {
52
+ const content = await actions.getPageContent(ctx, sessionId, tabId, {
53
+ maxLength: opts.maxContentLength,
54
+ });
55
+ item.contentSections = content.sections?.length ?? 0;
56
+ item.content = content;
57
+ }
58
+ // Close tab
59
+ await actions.closeTab(ctx, sessionId, tabId).catch(() => { });
60
+ return item;
61
+ }
62
+ catch (err) {
63
+ const errorCode = err.errorCode ?? '';
64
+ lastError = err.message || 'Unknown error';
65
+ // Always try to close the tab
66
+ if (tabId) {
67
+ await actions.closeTab(ctx, sessionId, tabId).catch(() => { });
68
+ }
69
+ if (token?.canceled) {
70
+ return { url, success: false, error: 'Canceled' };
71
+ }
72
+ // Only retry on retryable errors, and only on first attempt
73
+ const retryable = RETRYABLE_ERRORS.includes(errorCode)
74
+ || err.name === 'TimeoutError'
75
+ || (lastError?.includes('timeout') ?? false);
76
+ if (attempt === 0 && retryable) {
77
+ continue;
78
+ }
79
+ return { url, success: false, error: lastError };
80
+ }
81
+ }
82
+ // Should not reach here, but guard
83
+ return { url, success: false, error: lastError ?? 'Max retries exceeded' };
84
+ }
85
+ // ===== Main executor =====
86
+ export async function executeBatchExtract(ctx, sessionId, inputs, onProgress, token) {
87
+ const urls = inputs.urls;
88
+ if (!urls || urls.length === 0) {
89
+ return { summary: { total: 0, succeeded: 0, failed: 0 }, items: [] };
90
+ }
91
+ // Resolve extract options with defaults
92
+ const extractOpts = {
93
+ pageInfo: inputs.extract?.pageInfo ?? true,
94
+ content: inputs.extract?.content ?? true,
95
+ maxElements: inputs.extract?.maxElements ?? 50,
96
+ maxContentLength: inputs.extract?.maxContentLength ?? 4000,
97
+ };
98
+ // Validate all URLs upfront
99
+ const items = new Array(urls.length);
100
+ const validIndices = [];
101
+ for (let i = 0; i < urls.length; i++) {
102
+ if (token?.canceled) {
103
+ for (let j = i; j < urls.length; j++) {
104
+ items[j] = { url: urls[j], success: false, error: 'Canceled' };
105
+ }
106
+ break;
107
+ }
108
+ const check = await validateUrlAsync(urls[i], ctx.urlOpts);
109
+ if (check.valid) {
110
+ validIndices.push(i);
111
+ }
112
+ else {
113
+ items[i] = { url: urls[i], success: false, error: check.reason ?? 'Invalid URL' };
114
+ }
115
+ }
116
+ // Calculate effective concurrency
117
+ const concurrency = Math.min(inputs.concurrency ?? 3, MAX_CONCURRENCY, Math.max(validIndices.length, 1));
118
+ // Track progress
119
+ let doneCount = items.filter(Boolean).length;
120
+ if (onProgress && doneCount > 0)
121
+ onProgress(doneCount);
122
+ // Run extraction with sliding window
123
+ await slidingWindow(validIndices, concurrency, async (urlIndex) => {
124
+ if (token?.canceled) {
125
+ return;
126
+ }
127
+ items[urlIndex] = await extractSingleUrl(ctx, sessionId, urls[urlIndex], extractOpts, token);
128
+ doneCount++;
129
+ onProgress?.(doneCount);
130
+ });
131
+ // Mark untouched entries as canceled to keep summary consistent.
132
+ for (let i = 0; i < items.length; i++) {
133
+ if (!items[i]) {
134
+ items[i] = { url: urls[i], success: false, error: 'Canceled' };
135
+ doneCount++;
136
+ onProgress?.(doneCount);
137
+ }
138
+ }
139
+ // Aggregate summary
140
+ let succeeded = 0;
141
+ let failed = 0;
142
+ for (const item of items) {
143
+ if (item.success)
144
+ succeeded++;
145
+ else
146
+ failed++;
147
+ }
148
+ return {
149
+ summary: { total: urls.length, succeeded, failed },
150
+ items,
151
+ };
152
+ }
153
+ //# sourceMappingURL=batch-extract.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"batch-extract.js","sourceRoot":"","sources":["../../../src/task/templates/batch-extract.ts"],"names":[],"mappings":"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"}