arise-browser 0.1.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.
Files changed (148) hide show
  1. package/LICENSE +190 -0
  2. package/README.md +247 -0
  3. package/deploy/neko/CONTEXT.md +37 -0
  4. package/deploy/neko/arise-browser.service +13 -0
  5. package/deploy/neko/neko.yaml +12 -0
  6. package/deploy/neko/openbox.xml +763 -0
  7. package/deploy/neko/policies.json +28 -0
  8. package/deploy/neko/pulseaudio.pa +16 -0
  9. package/deploy/neko/setup.sh +308 -0
  10. package/deploy/neko/xorg.conf +118 -0
  11. package/dist/bin/arise-browser.d.ts +26 -0
  12. package/dist/bin/arise-browser.d.ts.map +1 -0
  13. package/dist/bin/arise-browser.js +224 -0
  14. package/dist/bin/arise-browser.js.map +1 -0
  15. package/dist/src/browser/action-executor.d.ts +98 -0
  16. package/dist/src/browser/action-executor.d.ts.map +1 -0
  17. package/dist/src/browser/action-executor.js +2726 -0
  18. package/dist/src/browser/action-executor.js.map +1 -0
  19. package/dist/src/browser/behavior-recorder.d.ts +61 -0
  20. package/dist/src/browser/behavior-recorder.d.ts.map +1 -0
  21. package/dist/src/browser/behavior-recorder.js +442 -0
  22. package/dist/src/browser/behavior-recorder.js.map +1 -0
  23. package/dist/src/browser/browser-session.d.ts +202 -0
  24. package/dist/src/browser/browser-session.d.ts.map +1 -0
  25. package/dist/src/browser/browser-session.js +1647 -0
  26. package/dist/src/browser/browser-session.js.map +1 -0
  27. package/dist/src/browser/config.d.ts +43 -0
  28. package/dist/src/browser/config.d.ts.map +1 -0
  29. package/dist/src/browser/config.js +59 -0
  30. package/dist/src/browser/config.js.map +1 -0
  31. package/dist/src/browser/page-snapshot.d.ts +38 -0
  32. package/dist/src/browser/page-snapshot.d.ts.map +1 -0
  33. package/dist/src/browser/page-snapshot.js +241 -0
  34. package/dist/src/browser/page-snapshot.js.map +1 -0
  35. package/dist/src/browser/scripts/behavior_tracker.js +424 -0
  36. package/dist/src/browser/scripts/unified_analyzer.js +1576 -0
  37. package/dist/src/index.d.ts +15 -0
  38. package/dist/src/index.d.ts.map +1 -0
  39. package/dist/src/index.js +15 -0
  40. package/dist/src/index.js.map +1 -0
  41. package/dist/src/lock.d.ts +11 -0
  42. package/dist/src/lock.d.ts.map +1 -0
  43. package/dist/src/lock.js +47 -0
  44. package/dist/src/lock.js.map +1 -0
  45. package/dist/src/logger.d.ts +17 -0
  46. package/dist/src/logger.d.ts.map +1 -0
  47. package/dist/src/logger.js +29 -0
  48. package/dist/src/logger.js.map +1 -0
  49. package/dist/src/server/middleware/auth.d.ts +6 -0
  50. package/dist/src/server/middleware/auth.d.ts.map +1 -0
  51. package/dist/src/server/middleware/auth.js +24 -0
  52. package/dist/src/server/middleware/auth.js.map +1 -0
  53. package/dist/src/server/route-utils.d.ts +15 -0
  54. package/dist/src/server/route-utils.d.ts.map +1 -0
  55. package/dist/src/server/route-utils.js +33 -0
  56. package/dist/src/server/route-utils.js.map +1 -0
  57. package/dist/src/server/routes/action.d.ts +5 -0
  58. package/dist/src/server/routes/action.d.ts.map +1 -0
  59. package/dist/src/server/routes/action.js +69 -0
  60. package/dist/src/server/routes/action.js.map +1 -0
  61. package/dist/src/server/routes/actions.d.ts +3 -0
  62. package/dist/src/server/routes/actions.d.ts.map +1 -0
  63. package/dist/src/server/routes/actions.js +53 -0
  64. package/dist/src/server/routes/actions.js.map +1 -0
  65. package/dist/src/server/routes/cookies.d.ts +3 -0
  66. package/dist/src/server/routes/cookies.d.ts.map +1 -0
  67. package/dist/src/server/routes/cookies.js +27 -0
  68. package/dist/src/server/routes/cookies.js.map +1 -0
  69. package/dist/src/server/routes/download.d.ts +3 -0
  70. package/dist/src/server/routes/download.d.ts.map +1 -0
  71. package/dist/src/server/routes/download.js +35 -0
  72. package/dist/src/server/routes/download.js.map +1 -0
  73. package/dist/src/server/routes/evaluate.d.ts +3 -0
  74. package/dist/src/server/routes/evaluate.d.ts.map +1 -0
  75. package/dist/src/server/routes/evaluate.js +27 -0
  76. package/dist/src/server/routes/evaluate.js.map +1 -0
  77. package/dist/src/server/routes/health.d.ts +3 -0
  78. package/dist/src/server/routes/health.d.ts.map +1 -0
  79. package/dist/src/server/routes/health.js +11 -0
  80. package/dist/src/server/routes/health.js.map +1 -0
  81. package/dist/src/server/routes/navigate.d.ts +3 -0
  82. package/dist/src/server/routes/navigate.d.ts.map +1 -0
  83. package/dist/src/server/routes/navigate.js +36 -0
  84. package/dist/src/server/routes/navigate.js.map +1 -0
  85. package/dist/src/server/routes/page-model.d.ts +3 -0
  86. package/dist/src/server/routes/page-model.d.ts.map +1 -0
  87. package/dist/src/server/routes/page-model.js +22 -0
  88. package/dist/src/server/routes/page-model.js.map +1 -0
  89. package/dist/src/server/routes/pdf.d.ts +3 -0
  90. package/dist/src/server/routes/pdf.d.ts.map +1 -0
  91. package/dist/src/server/routes/pdf.js +20 -0
  92. package/dist/src/server/routes/pdf.js.map +1 -0
  93. package/dist/src/server/routes/recording.d.ts +5 -0
  94. package/dist/src/server/routes/recording.d.ts.map +1 -0
  95. package/dist/src/server/routes/recording.js +217 -0
  96. package/dist/src/server/routes/recording.js.map +1 -0
  97. package/dist/src/server/routes/screenshot.d.ts +3 -0
  98. package/dist/src/server/routes/screenshot.d.ts.map +1 -0
  99. package/dist/src/server/routes/screenshot.js +32 -0
  100. package/dist/src/server/routes/screenshot.js.map +1 -0
  101. package/dist/src/server/routes/snapshot.d.ts +3 -0
  102. package/dist/src/server/routes/snapshot.d.ts.map +1 -0
  103. package/dist/src/server/routes/snapshot.js +454 -0
  104. package/dist/src/server/routes/snapshot.js.map +1 -0
  105. package/dist/src/server/routes/tab-lock.d.ts +3 -0
  106. package/dist/src/server/routes/tab-lock.d.ts.map +1 -0
  107. package/dist/src/server/routes/tab-lock.js +30 -0
  108. package/dist/src/server/routes/tab-lock.js.map +1 -0
  109. package/dist/src/server/routes/tab.d.ts +3 -0
  110. package/dist/src/server/routes/tab.d.ts.map +1 -0
  111. package/dist/src/server/routes/tab.js +47 -0
  112. package/dist/src/server/routes/tab.js.map +1 -0
  113. package/dist/src/server/routes/tabs.d.ts +3 -0
  114. package/dist/src/server/routes/tabs.d.ts.map +1 -0
  115. package/dist/src/server/routes/tabs.js +13 -0
  116. package/dist/src/server/routes/tabs.js.map +1 -0
  117. package/dist/src/server/routes/text.d.ts +3 -0
  118. package/dist/src/server/routes/text.d.ts.map +1 -0
  119. package/dist/src/server/routes/text.js +20 -0
  120. package/dist/src/server/routes/text.js.map +1 -0
  121. package/dist/src/server/routes/upload.d.ts +3 -0
  122. package/dist/src/server/routes/upload.d.ts.map +1 -0
  123. package/dist/src/server/routes/upload.js +38 -0
  124. package/dist/src/server/routes/upload.js.map +1 -0
  125. package/dist/src/server/server.d.ts +7 -0
  126. package/dist/src/server/server.d.ts.map +1 -0
  127. package/dist/src/server/server.js +69 -0
  128. package/dist/src/server/server.js.map +1 -0
  129. package/dist/src/types/index.d.ts +125 -0
  130. package/dist/src/types/index.d.ts.map +1 -0
  131. package/dist/src/types/index.js +5 -0
  132. package/dist/src/types/index.js.map +1 -0
  133. package/dist/src/virtual-display/manager.d.ts +37 -0
  134. package/dist/src/virtual-display/manager.d.ts.map +1 -0
  135. package/dist/src/virtual-display/manager.js +229 -0
  136. package/dist/src/virtual-display/manager.js.map +1 -0
  137. package/dist/src/virtual-display/process-runner.d.ts +43 -0
  138. package/dist/src/virtual-display/process-runner.d.ts.map +1 -0
  139. package/dist/src/virtual-display/process-runner.js +174 -0
  140. package/dist/src/virtual-display/process-runner.js.map +1 -0
  141. package/dist/tsconfig.tsbuildinfo +1 -0
  142. package/package.json +57 -0
  143. package/plugin/openclaw.plugin.json +148 -0
  144. package/skill/arise-browser/SKILL.md +275 -0
  145. package/skill/arise-browser/TRUST.md +42 -0
  146. package/skill/arise-browser/references/api.md +198 -0
  147. package/src/browser/scripts/behavior_tracker.js +424 -0
  148. package/src/browser/scripts/unified_analyzer.js +1576 -0
@@ -0,0 +1,36 @@
1
+ import { getTabWriteConflict, sendRouteError, sendTabLocked } from "../route-utils.js";
2
+ export function registerNavigateRoute(app) {
3
+ app.post("/navigate", async (request, reply) => {
4
+ const session = app.session;
5
+ const { url, newTab, timeout, tabId, owner } = request.body || {};
6
+ if (!url) {
7
+ return reply.code(400).send({ error: "url is required" });
8
+ }
9
+ if (timeout !== undefined && (!Number.isFinite(timeout) || timeout <= 0)) {
10
+ return reply.code(400).send({ error: "timeout must be a positive number" });
11
+ }
12
+ try {
13
+ if (newTab) {
14
+ const [newTabId] = await session.createNewTab(url, { timeout });
15
+ await session.switchToTab(newTabId);
16
+ const info = await session.getPageInfo(newTabId);
17
+ return { tabId: newTabId, url: info.url || url, title: info.title };
18
+ }
19
+ const conflict = getTabWriteConflict(session, { tabId, owner });
20
+ if (conflict) {
21
+ return sendTabLocked(reply, conflict);
22
+ }
23
+ await session.visit(url, { tabId, timeout });
24
+ const info = await session.getPageInfo(tabId);
25
+ return {
26
+ tabId: info.tabId,
27
+ url: info.url || url,
28
+ title: info.title,
29
+ };
30
+ }
31
+ catch (e) {
32
+ return sendRouteError(reply, e, "Navigation failed");
33
+ }
34
+ });
35
+ }
36
+ //# sourceMappingURL=navigate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"navigate.js","sourceRoot":"","sources":["../../../../src/server/routes/navigate.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,mBAAmB,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAUvF,MAAM,UAAU,qBAAqB,CAAC,GAAoB;IACxD,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,EAAE,OAA+C,EAAE,KAAK,EAAE,EAAE;QACrF,MAAM,OAAO,GAAI,GAAW,CAAC,OAAyB,CAAC;QACvD,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,OAAO,CAAC,IAAI,IAAI,EAAkB,CAAC;QAElF,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,CAAC,CAAC;QAC5D,CAAC;QAED,IAAI,OAAO,KAAK,SAAS,IAAI,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,OAAO,IAAI,CAAC,CAAC,EAAE,CAAC;YACzE,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mCAAmC,EAAE,CAAC,CAAC;QAC9E,CAAC;QAED,IAAI,CAAC;YACH,IAAI,MAAM,EAAE,CAAC;gBACX,MAAM,CAAC,QAAQ,CAAC,GAAG,MAAM,OAAO,CAAC,YAAY,CAAC,GAAG,EAAE,EAAE,OAAO,EAAE,CAAC,CAAC;gBAChE,MAAM,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;gBACpC,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;gBACjD,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,IAAI,GAAG,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC;YACtE,CAAC;YAED,MAAM,QAAQ,GAAG,mBAAmB,CAAC,OAAO,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC;YAChE,IAAI,QAAQ,EAAE,CAAC;gBACb,OAAO,aAAa,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;YACxC,CAAC;YAED,MAAM,OAAO,CAAC,KAAK,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;YAC7C,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAC9C,OAAO;gBACL,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,GAAG,EAAE,IAAI,CAAC,GAAG,IAAI,GAAG;gBACpB,KAAK,EAAE,IAAI,CAAC,KAAK;aAClB,CAAC;QACJ,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,cAAc,CAAC,KAAK,EAAE,CAAC,EAAE,mBAAmB,CAAC,CAAC;QACvD,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerPageModelRoute(app: FastifyInstance): void;
3
+ //# sourceMappingURL=page-model.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"page-model.d.ts","sourceRoot":"","sources":["../../../../src/server/routes/page-model.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAkB,MAAM,SAAS,CAAC;AAS/D,wBAAgB,sBAAsB,CAAC,GAAG,EAAE,eAAe,QAqB1D"}
@@ -0,0 +1,22 @@
1
+ import { sendRouteError } from "../route-utils.js";
2
+ export function registerPageModelRoute(app) {
3
+ app.get("/page-model", async (request, reply) => {
4
+ const session = app.session;
5
+ const { tabId, includeRawText } = request.query;
6
+ try {
7
+ const model = await session.getPageModel(tabId, {
8
+ includeRawText: includeRawText === "true" || includeRawText === "1",
9
+ });
10
+ const info = await session.getPageInfo(tabId);
11
+ return {
12
+ model,
13
+ url: info.url,
14
+ title: info.title,
15
+ };
16
+ }
17
+ catch (e) {
18
+ return sendRouteError(reply, e, "Failed to build page model");
19
+ }
20
+ });
21
+ }
22
+ //# sourceMappingURL=page-model.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"page-model.js","sourceRoot":"","sources":["../../../../src/server/routes/page-model.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAOnD,MAAM,UAAU,sBAAsB,CAAC,GAAoB;IACzD,GAAG,CAAC,GAAG,CACL,aAAa,EACb,KAAK,EAAE,OAAwD,EAAE,KAAK,EAAE,EAAE;QACxE,MAAM,OAAO,GAAI,GAAW,CAAC,OAAyB,CAAC;QACvD,MAAM,EAAE,KAAK,EAAE,cAAc,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC;QAChD,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,MAAM,OAAO,CAAC,YAAY,CAAC,KAAK,EAAE;gBAC9C,cAAc,EAAE,cAAc,KAAK,MAAM,IAAI,cAAc,KAAK,GAAG;aACpE,CAAC,CAAC;YACH,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YAC9C,OAAO;gBACL,KAAK;gBACL,GAAG,EAAE,IAAI,CAAC,GAAG;gBACb,KAAK,EAAE,IAAI,CAAC,KAAK;aAClB,CAAC;QACJ,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,cAAc,CAAC,KAAK,EAAE,CAAC,EAAE,4BAA4B,CAAC,CAAC;QAChE,CAAC;IACH,CAAC,CACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerPdfRoute(app: FastifyInstance): void;
3
+ //# sourceMappingURL=pdf.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pdf.d.ts","sourceRoot":"","sources":["../../../../src/server/routes/pdf.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAkB,MAAM,SAAS,CAAC;AAQ/D,wBAAgB,gBAAgB,CAAC,GAAG,EAAE,eAAe,QAmBpD"}
@@ -0,0 +1,20 @@
1
+ import { sendRouteError } from "../route-utils.js";
2
+ export function registerPdfRoute(app) {
3
+ app.get("/pdf", async (request, reply) => {
4
+ const session = app.session;
5
+ const { tabId } = request.query;
6
+ try {
7
+ const buffer = await session.exportPdf(tabId);
8
+ if (!buffer) {
9
+ return reply.code(500).send({ error: "PDF export failed" });
10
+ }
11
+ return reply
12
+ .type("application/pdf")
13
+ .send(buffer);
14
+ }
15
+ catch (e) {
16
+ return sendRouteError(reply, e, "PDF export failed", 400);
17
+ }
18
+ });
19
+ }
20
+ //# sourceMappingURL=pdf.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pdf.js","sourceRoot":"","sources":["../../../../src/server/routes/pdf.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAMnD,MAAM,UAAU,gBAAgB,CAAC,GAAoB;IACnD,GAAG,CAAC,GAAG,CAAC,MAAM,EAAE,KAAK,EAAE,OAAkD,EAAE,KAAK,EAAE,EAAE;QAClF,MAAM,OAAO,GAAI,GAAW,CAAC,OAAyB,CAAC;QACvD,MAAM,EAAE,KAAK,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC;QAEhC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAE9C,IAAI,CAAC,MAAM,EAAE,CAAC;gBACZ,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC,CAAC;YAC9D,CAAC;YAED,OAAO,KAAK;iBACT,IAAI,CAAC,iBAAiB,CAAC;iBACvB,IAAI,CAAC,MAAM,CAAC,CAAC;QAClB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,cAAc,CAAC,KAAK,EAAE,CAAC,EAAE,mBAAmB,EAAE,GAAG,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ /** Stop and remove all active recorders (called on server shutdown). */
3
+ export declare function cleanupRecorders(): Promise<void>;
4
+ export declare function registerRecordingRoutes(app: FastifyInstance): void;
5
+ //# sourceMappingURL=recording.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recording.d.ts","sourceRoot":"","sources":["../../../../src/server/routes/recording.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAkB,MAAM,SAAS,CAAC;AA0H/D,wEAAwE;AACxE,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,IAAI,CAAC,CAYtD;AAED,wBAAgB,uBAAuB,CAAC,GAAG,EAAE,eAAe,QAmH3D"}
@@ -0,0 +1,217 @@
1
+ import { BehaviorRecorder } from "../../browser/behavior-recorder.js";
2
+ // Active recorders by recording ID
3
+ const activeRecorders = new Map();
4
+ const completedRecordings = new Map();
5
+ const MAX_RECORDERS = 10;
6
+ const MAX_COMPLETED_RECORDINGS = 100;
7
+ const COMPLETED_TTL_MS = 60 * 60 * 1000;
8
+ function pruneCompletedRecordings(now = Date.now()) {
9
+ for (const [recordingId, completed] of completedRecordings) {
10
+ if (completed.completedAt + COMPLETED_TTL_MS < now) {
11
+ completedRecordings.delete(recordingId);
12
+ }
13
+ }
14
+ if (completedRecordings.size <= MAX_COMPLETED_RECORDINGS) {
15
+ return;
16
+ }
17
+ const oldestFirst = [...completedRecordings.entries()]
18
+ .sort((a, b) => a[1].completedAt - b[1].completedAt);
19
+ const overflow = completedRecordings.size - MAX_COMPLETED_RECORDINGS;
20
+ for (const [recordingId] of oldestFirst.slice(0, overflow)) {
21
+ completedRecordings.delete(recordingId);
22
+ }
23
+ }
24
+ function getCompletedRecording(recordingId) {
25
+ pruneCompletedRecordings();
26
+ return completedRecordings.get(recordingId) ?? null;
27
+ }
28
+ function storeCompletedRecording(recordingId, result) {
29
+ pruneCompletedRecordings();
30
+ completedRecordings.set(recordingId, {
31
+ result,
32
+ completedAt: Date.now(),
33
+ });
34
+ pruneCompletedRecordings();
35
+ }
36
+ function buildLearnData(operations, task) {
37
+ const steps = [];
38
+ const domains = new Set();
39
+ let startTime = null;
40
+ let endTime = null;
41
+ const pageUrls = new Set();
42
+ for (const op of operations) {
43
+ const ts = new Date(op.timestamp).getTime();
44
+ if (startTime === null || ts < startTime)
45
+ startTime = ts;
46
+ if (endTime === null || ts > endTime)
47
+ endTime = ts;
48
+ if (op.url) {
49
+ pageUrls.add(op.url);
50
+ try {
51
+ domains.add(new URL(op.url).hostname);
52
+ }
53
+ catch {
54
+ // invalid URL
55
+ }
56
+ }
57
+ if (op.type === "navigate" && op.url) {
58
+ steps.push({ url: op.url, action: "navigate" });
59
+ }
60
+ else if (op.type === "click") {
61
+ steps.push({
62
+ url: op.url || "",
63
+ action: "click",
64
+ target: op.text || op.ref,
65
+ });
66
+ }
67
+ else if (op.type === "type") {
68
+ steps.push({
69
+ url: op.url || "",
70
+ action: "type",
71
+ target: op.ref,
72
+ value: op.value,
73
+ });
74
+ }
75
+ else if (op.type === "scroll") {
76
+ steps.push({
77
+ url: op.url || "",
78
+ action: "scroll",
79
+ });
80
+ }
81
+ else if (op.type === "select") {
82
+ steps.push({
83
+ url: op.url || "",
84
+ action: "select",
85
+ target: op.ref,
86
+ value: op.value,
87
+ });
88
+ }
89
+ }
90
+ return {
91
+ type: "browser_workflow",
92
+ task: task || "untitled recording",
93
+ success: true,
94
+ source: "arise-browser",
95
+ domain: [...domains][0] || "",
96
+ steps,
97
+ metadata: {
98
+ duration_ms: startTime !== null && endTime !== null ? endTime - startTime : 0,
99
+ page_count: pageUrls.size,
100
+ recorded_at: new Date().toISOString(),
101
+ },
102
+ };
103
+ }
104
+ /** Stop and remove all active recorders (called on server shutdown). */
105
+ export async function cleanupRecorders() {
106
+ for (const [, recorder] of activeRecorders) {
107
+ try {
108
+ if (recorder.isRecording()) {
109
+ await recorder.stopRecording();
110
+ }
111
+ }
112
+ catch {
113
+ // best effort
114
+ }
115
+ }
116
+ activeRecorders.clear();
117
+ completedRecordings.clear();
118
+ }
119
+ export function registerRecordingRoutes(app) {
120
+ // Cleanup recorders on server close
121
+ app.addHook("onClose", async () => {
122
+ await cleanupRecorders();
123
+ });
124
+ app.post("/recording/start", async (_request, reply) => {
125
+ const session = app.session;
126
+ pruneCompletedRecordings();
127
+ if (activeRecorders.size >= MAX_RECORDERS) {
128
+ return reply.code(429).send({ error: `Maximum ${MAX_RECORDERS} concurrent recordings reached` });
129
+ }
130
+ try {
131
+ const recorder = new BehaviorRecorder(true);
132
+ await recorder.startRecording(session);
133
+ activeRecorders.set(recorder.sessionId, recorder);
134
+ return { recordingId: recorder.sessionId };
135
+ }
136
+ catch {
137
+ return reply.code(500).send({ error: "Failed to start recording" });
138
+ }
139
+ });
140
+ app.post("/recording/stop", async (request, reply) => {
141
+ const { recordingId } = request.body || {};
142
+ if (!recordingId) {
143
+ return reply.code(400).send({ error: "recordingId is required" });
144
+ }
145
+ const completed = getCompletedRecording(recordingId);
146
+ if (completed) {
147
+ return completed.result;
148
+ }
149
+ const recorder = activeRecorders.get(recordingId);
150
+ if (!recorder) {
151
+ return reply.code(404).send({ error: "Recording not found" });
152
+ }
153
+ try {
154
+ const result = await recorder.stopRecording();
155
+ activeRecorders.delete(recordingId);
156
+ storeCompletedRecording(recordingId, result);
157
+ return result;
158
+ }
159
+ catch {
160
+ activeRecorders.delete(recordingId);
161
+ return reply.code(500).send({ error: "Failed to stop recording" });
162
+ }
163
+ });
164
+ app.get("/recording/status", async (request) => {
165
+ const { recordingId } = request.query;
166
+ pruneCompletedRecordings();
167
+ if (recordingId) {
168
+ const recorder = activeRecorders.get(recordingId);
169
+ if (recorder) {
170
+ return {
171
+ active: recorder.isRecording(),
172
+ count: recorder.getOperationsCount(),
173
+ recordingId,
174
+ };
175
+ }
176
+ const completed = getCompletedRecording(recordingId);
177
+ if (!completed) {
178
+ return { active: false, completed: false, count: 0 };
179
+ }
180
+ return {
181
+ active: false,
182
+ completed: true,
183
+ count: completed.result.operations_count,
184
+ recordingId,
185
+ };
186
+ }
187
+ const active = [];
188
+ for (const [id, recorder] of activeRecorders) {
189
+ active.push({
190
+ recordingId: id,
191
+ active: recorder.isRecording(),
192
+ count: recorder.getOperationsCount(),
193
+ });
194
+ }
195
+ return {
196
+ recordings: active,
197
+ completed_count: completedRecordings.size,
198
+ };
199
+ });
200
+ app.post("/recording/export", async (request, reply) => {
201
+ const { recordingId, task } = request.body || {};
202
+ if (!recordingId) {
203
+ return reply.code(400).send({ error: "recordingId is required" });
204
+ }
205
+ pruneCompletedRecordings();
206
+ const recorder = activeRecorders.get(recordingId);
207
+ const completed = getCompletedRecording(recordingId);
208
+ if (!recorder && !completed) {
209
+ return reply.code(404).send({ error: "Recording not found" });
210
+ }
211
+ const operations = recorder
212
+ ? recorder.getOperations()
213
+ : completed.result.operations;
214
+ return buildLearnData(operations, task);
215
+ });
216
+ }
217
+ //# sourceMappingURL=recording.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recording.js","sourceRoot":"","sources":["../../../../src/server/routes/recording.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerScreenshotRoute(app: FastifyInstance): void;
3
+ //# sourceMappingURL=screenshot.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"screenshot.d.ts","sourceRoot":"","sources":["../../../../src/server/routes/screenshot.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAkB,MAAM,SAAS,CAAC;AAU/D,wBAAgB,uBAAuB,CAAC,GAAG,EAAE,eAAe,QAgC3D"}
@@ -0,0 +1,32 @@
1
+ import { sendRouteError } from "../route-utils.js";
2
+ export function registerScreenshotRoute(app) {
3
+ app.get("/screenshot", async (request, reply) => {
4
+ const session = app.session;
5
+ const { tabId, quality = "75", raw } = request.query;
6
+ try {
7
+ const buffer = await session.takeScreenshot({
8
+ tabId,
9
+ type: "jpeg",
10
+ quality: Math.min(100, Math.max(1, parseInt(quality) || 75)),
11
+ });
12
+ if (!buffer) {
13
+ return reply.code(500).send({ error: "Screenshot failed" });
14
+ }
15
+ const acceptHeader = request.headers.accept || "";
16
+ const wantsRaw = raw === "true" || raw === "1" || acceptHeader.includes("image/");
17
+ if (wantsRaw) {
18
+ return reply
19
+ .type("image/jpeg")
20
+ .send(buffer);
21
+ }
22
+ return {
23
+ image: `data:image/jpeg;base64,${buffer.toString("base64")}`,
24
+ format: "jpeg",
25
+ };
26
+ }
27
+ catch (e) {
28
+ return sendRouteError(reply, e, "Screenshot failed");
29
+ }
30
+ });
31
+ }
32
+ //# sourceMappingURL=screenshot.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"screenshot.js","sourceRoot":"","sources":["../../../../src/server/routes/screenshot.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAQnD,MAAM,UAAU,uBAAuB,CAAC,GAAoB;IAC1D,GAAG,CAAC,GAAG,CAAC,aAAa,EAAE,KAAK,EAAE,OAAyD,EAAE,KAAK,EAAE,EAAE;QAChG,MAAM,OAAO,GAAI,GAAW,CAAC,OAAyB,CAAC;QACvD,MAAM,EAAE,KAAK,EAAE,OAAO,GAAG,IAAI,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC;QAErD,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,cAAc,CAAC;gBAC1C,KAAK;gBACL,IAAI,EAAE,MAAM;gBACZ,OAAO,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;aAC7D,CAAC,CAAC;YAEH,IAAI,CAAC,MAAM,EAAE,CAAC;gBACZ,OAAO,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC,CAAC;YAC9D,CAAC;YAED,MAAM,YAAY,GAAG,OAAO,CAAC,OAAO,CAAC,MAAM,IAAI,EAAE,CAAC;YAClD,MAAM,QAAQ,GAAG,GAAG,KAAK,MAAM,IAAI,GAAG,KAAK,GAAG,IAAI,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;YAClF,IAAI,QAAQ,EAAE,CAAC;gBACb,OAAO,KAAK;qBACT,IAAI,CAAC,YAAY,CAAC;qBAClB,IAAI,CAAC,MAAM,CAAC,CAAC;YAClB,CAAC;YAED,OAAO;gBACL,KAAK,EAAE,0BAA0B,MAAM,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE;gBAC5D,MAAM,EAAE,MAAM;aACf,CAAC;QACJ,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,cAAc,CAAC,KAAK,EAAE,CAAC,EAAE,mBAAmB,CAAC,CAAC;QACvD,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { FastifyInstance } from "fastify";
2
+ export declare function registerSnapshotRoute(app: FastifyInstance): void;
3
+ //# sourceMappingURL=snapshot.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"snapshot.d.ts","sourceRoot":"","sources":["../../../../src/server/routes/snapshot.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAkB,MAAM,SAAS,CAAC;AAke/D,wBAAgB,qBAAqB,CAAC,GAAG,EAAE,eAAe,QAuFzD"}