explorbot 0.0.1

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 (329) hide show
  1. package/LICENSE +94 -0
  2. package/README.md +267 -0
  3. package/assets/sample-files/sample.docx +0 -0
  4. package/assets/sample-files/sample.mp3 +0 -0
  5. package/assets/sample-files/sample.mp4 +0 -0
  6. package/assets/sample-files/sample.pdf +21 -0
  7. package/assets/sample-files/sample.png +0 -0
  8. package/assets/sample-files/sample.xlsx +0 -0
  9. package/assets/sample-files/sample.zip +0 -0
  10. package/dist/assets/sample-files/sample.docx +0 -0
  11. package/dist/assets/sample-files/sample.mp3 +0 -0
  12. package/dist/assets/sample-files/sample.mp4 +0 -0
  13. package/dist/assets/sample-files/sample.pdf +21 -0
  14. package/dist/assets/sample-files/sample.png +0 -0
  15. package/dist/assets/sample-files/sample.xlsx +0 -0
  16. package/dist/assets/sample-files/sample.zip +0 -0
  17. package/dist/bin/explorbot-cli.js +683 -0
  18. package/dist/bin/explorbot-cli.js.map +1 -0
  19. package/dist/boat/api-tester/bin/apibot-cli.js +5 -0
  20. package/dist/boat/api-tester/bin/apibot-cli.js.map +1 -0
  21. package/dist/boat/api-tester/example/apibot.config.js +31 -0
  22. package/dist/boat/api-tester/example/apibot.config.js.map +1 -0
  23. package/dist/boat/api-tester/src/ai/chief/styles.js +13 -0
  24. package/dist/boat/api-tester/src/ai/chief/styles.js.map +1 -0
  25. package/dist/boat/api-tester/src/ai/chief.js +301 -0
  26. package/dist/boat/api-tester/src/ai/chief.js.map +1 -0
  27. package/dist/boat/api-tester/src/ai/curler-tools.js +263 -0
  28. package/dist/boat/api-tester/src/ai/curler-tools.js.map +1 -0
  29. package/dist/boat/api-tester/src/ai/curler.js +271 -0
  30. package/dist/boat/api-tester/src/ai/curler.js.map +1 -0
  31. package/dist/boat/api-tester/src/api-client.js +26 -0
  32. package/dist/boat/api-tester/src/api-client.js.map +1 -0
  33. package/dist/boat/api-tester/src/apibot.js +166 -0
  34. package/dist/boat/api-tester/src/apibot.js.map +1 -0
  35. package/dist/boat/api-tester/src/cli.js +262 -0
  36. package/dist/boat/api-tester/src/cli.js.map +1 -0
  37. package/dist/boat/api-tester/src/config.js +159 -0
  38. package/dist/boat/api-tester/src/config.js.map +1 -0
  39. package/dist/prompts/audit-rules.md +124 -0
  40. package/dist/rules/chief/general.md +11 -0
  41. package/dist/rules/chief/styles/curious.md +12 -0
  42. package/dist/rules/chief/styles/hacker.md +19 -0
  43. package/dist/rules/chief/styles/normal.md +11 -0
  44. package/dist/rules/chief/styles/psycho.md +17 -0
  45. package/dist/rules/navigator/multiple-locator.md +47 -0
  46. package/dist/rules/navigator/output.md +69 -0
  47. package/dist/rules/navigator/verification-actions.md +122 -0
  48. package/dist/rules/navigator/verification-output.md +53 -0
  49. package/dist/rules/planner/styles/curious.md +39 -0
  50. package/dist/rules/planner/styles/normal.md +21 -0
  51. package/dist/rules/planner/styles/psycho.md +14 -0
  52. package/dist/rules/researcher/list-element.md +11 -0
  53. package/dist/rules/researcher/screenshot-ui-map.md +30 -0
  54. package/dist/rules/researcher/section-ui-map.md +18 -0
  55. package/dist/rules/researcher/ui-map-table.md +18 -0
  56. package/dist/src/action-result.js +574 -0
  57. package/dist/src/action-result.js.map +1 -0
  58. package/dist/src/action.js +388 -0
  59. package/dist/src/action.js.map +1 -0
  60. package/dist/src/activity.js +86 -0
  61. package/dist/src/activity.js.map +1 -0
  62. package/dist/src/ai/agent.js +2 -0
  63. package/dist/src/ai/agent.js.map +1 -0
  64. package/dist/src/ai/bosun.js +443 -0
  65. package/dist/src/ai/bosun.js.map +1 -0
  66. package/dist/src/ai/captain/idle-mode.js +102 -0
  67. package/dist/src/ai/captain/idle-mode.js.map +1 -0
  68. package/dist/src/ai/captain/mixin.js +11 -0
  69. package/dist/src/ai/captain/mixin.js.map +1 -0
  70. package/dist/src/ai/captain/test-mode.js +251 -0
  71. package/dist/src/ai/captain/test-mode.js.map +1 -0
  72. package/dist/src/ai/captain/web-mode.js +124 -0
  73. package/dist/src/ai/captain/web-mode.js.map +1 -0
  74. package/dist/src/ai/captain.js +442 -0
  75. package/dist/src/ai/captain.js.map +1 -0
  76. package/dist/src/ai/conversation.js +176 -0
  77. package/dist/src/ai/conversation.js.map +1 -0
  78. package/dist/src/ai/experience-compactor.js +232 -0
  79. package/dist/src/ai/experience-compactor.js.map +1 -0
  80. package/dist/src/ai/fisherman-tools.js +154 -0
  81. package/dist/src/ai/fisherman-tools.js.map +1 -0
  82. package/dist/src/ai/fisherman.js +184 -0
  83. package/dist/src/ai/fisherman.js.map +1 -0
  84. package/dist/src/ai/historian.js +384 -0
  85. package/dist/src/ai/historian.js.map +1 -0
  86. package/dist/src/ai/navigator.js +493 -0
  87. package/dist/src/ai/navigator.js.map +1 -0
  88. package/dist/src/ai/pilot.js +684 -0
  89. package/dist/src/ai/pilot.js.map +1 -0
  90. package/dist/src/ai/planner/session-dedup.js +28 -0
  91. package/dist/src/ai/planner/session-dedup.js.map +1 -0
  92. package/dist/src/ai/planner/styles.js +15 -0
  93. package/dist/src/ai/planner/styles.js.map +1 -0
  94. package/dist/src/ai/planner/subpages.js +118 -0
  95. package/dist/src/ai/planner/subpages.js.map +1 -0
  96. package/dist/src/ai/planner.js +486 -0
  97. package/dist/src/ai/planner.js.map +1 -0
  98. package/dist/src/ai/provider.js +540 -0
  99. package/dist/src/ai/provider.js.map +1 -0
  100. package/dist/src/ai/quartermaster.js +210 -0
  101. package/dist/src/ai/quartermaster.js.map +1 -0
  102. package/dist/src/ai/researcher/cache.js +95 -0
  103. package/dist/src/ai/researcher/cache.js.map +1 -0
  104. package/dist/src/ai/researcher/coordinates.js +210 -0
  105. package/dist/src/ai/researcher/coordinates.js.map +1 -0
  106. package/dist/src/ai/researcher/deep-analysis.js +364 -0
  107. package/dist/src/ai/researcher/deep-analysis.js.map +1 -0
  108. package/dist/src/ai/researcher/fingerprint-worker.js +46 -0
  109. package/dist/src/ai/researcher/fingerprint-worker.js.map +1 -0
  110. package/dist/src/ai/researcher/focus.js +37 -0
  111. package/dist/src/ai/researcher/focus.js.map +1 -0
  112. package/dist/src/ai/researcher/locators.js +242 -0
  113. package/dist/src/ai/researcher/locators.js.map +1 -0
  114. package/dist/src/ai/researcher/mixin.js +3 -0
  115. package/dist/src/ai/researcher/mixin.js.map +1 -0
  116. package/dist/src/ai/researcher/parser.js +160 -0
  117. package/dist/src/ai/researcher/parser.js.map +1 -0
  118. package/dist/src/ai/researcher/research-result.js +110 -0
  119. package/dist/src/ai/researcher/research-result.js.map +1 -0
  120. package/dist/src/ai/researcher.js +776 -0
  121. package/dist/src/ai/researcher.js.map +1 -0
  122. package/dist/src/ai/rules.js +368 -0
  123. package/dist/src/ai/rules.js.map +1 -0
  124. package/dist/src/ai/task-agent.js +110 -0
  125. package/dist/src/ai/task-agent.js.map +1 -0
  126. package/dist/src/ai/tester.js +840 -0
  127. package/dist/src/ai/tester.js.map +1 -0
  128. package/dist/src/ai/tools.js +980 -0
  129. package/dist/src/ai/tools.js.map +1 -0
  130. package/dist/src/api/api-client.js +91 -0
  131. package/dist/src/api/api-client.js.map +1 -0
  132. package/dist/src/api/request-result.js +177 -0
  133. package/dist/src/api/request-result.js.map +1 -0
  134. package/dist/src/api/request-store.js +109 -0
  135. package/dist/src/api/request-store.js.map +1 -0
  136. package/dist/src/api/spec-reader.js +148 -0
  137. package/dist/src/api/spec-reader.js.map +1 -0
  138. package/dist/src/api/xhr-capture.js +91 -0
  139. package/dist/src/api/xhr-capture.js.map +1 -0
  140. package/dist/src/browser-server.js +67 -0
  141. package/dist/src/browser-server.js.map +1 -0
  142. package/dist/src/command-handler.js +363 -0
  143. package/dist/src/command-handler.js.map +1 -0
  144. package/dist/src/commands/add-rule-command.js +52 -0
  145. package/dist/src/commands/add-rule-command.js.map +1 -0
  146. package/dist/src/commands/base-command.js +14 -0
  147. package/dist/src/commands/base-command.js.map +1 -0
  148. package/dist/src/commands/clean-command.js +67 -0
  149. package/dist/src/commands/clean-command.js.map +1 -0
  150. package/dist/src/commands/context-aria-command.js +18 -0
  151. package/dist/src/commands/context-aria-command.js.map +1 -0
  152. package/dist/src/commands/context-command.js +57 -0
  153. package/dist/src/commands/context-command.js.map +1 -0
  154. package/dist/src/commands/context-data-command.js +25 -0
  155. package/dist/src/commands/context-data-command.js.map +1 -0
  156. package/dist/src/commands/context-experience-command.js +41 -0
  157. package/dist/src/commands/context-experience-command.js.map +1 -0
  158. package/dist/src/commands/context-html-command.js +26 -0
  159. package/dist/src/commands/context-html-command.js.map +1 -0
  160. package/dist/src/commands/context-knowledge-command.js +36 -0
  161. package/dist/src/commands/context-knowledge-command.js.map +1 -0
  162. package/dist/src/commands/debug-command.js +12 -0
  163. package/dist/src/commands/debug-command.js.map +1 -0
  164. package/dist/src/commands/drill-command.js +29 -0
  165. package/dist/src/commands/drill-command.js.map +1 -0
  166. package/dist/src/commands/exit-command.js +26 -0
  167. package/dist/src/commands/exit-command.js.map +1 -0
  168. package/dist/src/commands/explore-command.js +124 -0
  169. package/dist/src/commands/explore-command.js.map +1 -0
  170. package/dist/src/commands/freesail-command.js +84 -0
  171. package/dist/src/commands/freesail-command.js.map +1 -0
  172. package/dist/src/commands/help-command.js +7 -0
  173. package/dist/src/commands/help-command.js.map +1 -0
  174. package/dist/src/commands/index.js +63 -0
  175. package/dist/src/commands/index.js.map +1 -0
  176. package/dist/src/commands/knows-command.js +54 -0
  177. package/dist/src/commands/knows-command.js.map +1 -0
  178. package/dist/src/commands/learn-command.js +35 -0
  179. package/dist/src/commands/learn-command.js.map +1 -0
  180. package/dist/src/commands/navigate-command.js +16 -0
  181. package/dist/src/commands/navigate-command.js.map +1 -0
  182. package/dist/src/commands/path-command.js +70 -0
  183. package/dist/src/commands/path-command.js.map +1 -0
  184. package/dist/src/commands/plan-clear-command.js +13 -0
  185. package/dist/src/commands/plan-clear-command.js.map +1 -0
  186. package/dist/src/commands/plan-command.js +36 -0
  187. package/dist/src/commands/plan-command.js.map +1 -0
  188. package/dist/src/commands/plan-edit-command.js +8 -0
  189. package/dist/src/commands/plan-edit-command.js.map +1 -0
  190. package/dist/src/commands/plan-load-command.js +16 -0
  191. package/dist/src/commands/plan-load-command.js.map +1 -0
  192. package/dist/src/commands/plan-reload-command.js +23 -0
  193. package/dist/src/commands/plan-reload-command.js.map +1 -0
  194. package/dist/src/commands/plan-save-command.js +22 -0
  195. package/dist/src/commands/plan-save-command.js.map +1 -0
  196. package/dist/src/commands/research-command.js +38 -0
  197. package/dist/src/commands/research-command.js.map +1 -0
  198. package/dist/src/commands/start-command.js +12 -0
  199. package/dist/src/commands/start-command.js.map +1 -0
  200. package/dist/src/commands/status-command.js +19 -0
  201. package/dist/src/commands/status-command.js.map +1 -0
  202. package/dist/src/commands/test-command.js +85 -0
  203. package/dist/src/commands/test-command.js.map +1 -0
  204. package/dist/src/components/ActivityPane.js +55 -0
  205. package/dist/src/components/ActivityPane.js.map +1 -0
  206. package/dist/src/components/AddKnowledge.js +122 -0
  207. package/dist/src/components/AddKnowledge.js.map +1 -0
  208. package/dist/src/components/AddRule.js +117 -0
  209. package/dist/src/components/AddRule.js.map +1 -0
  210. package/dist/src/components/App.js +313 -0
  211. package/dist/src/components/App.js.map +1 -0
  212. package/dist/src/components/Autocomplete.js +43 -0
  213. package/dist/src/components/Autocomplete.js.map +1 -0
  214. package/dist/src/components/InputPane.js +207 -0
  215. package/dist/src/components/InputPane.js.map +1 -0
  216. package/dist/src/components/InputReadline.js +598 -0
  217. package/dist/src/components/InputReadline.js.map +1 -0
  218. package/dist/src/components/LogPane.js +123 -0
  219. package/dist/src/components/LogPane.js.map +1 -0
  220. package/dist/src/components/PlanEditor.js +126 -0
  221. package/dist/src/components/PlanEditor.js.map +1 -0
  222. package/dist/src/components/PlanPane.js +51 -0
  223. package/dist/src/components/PlanPane.js.map +1 -0
  224. package/dist/src/components/SessionTimer.js +26 -0
  225. package/dist/src/components/SessionTimer.js.map +1 -0
  226. package/dist/src/components/StateTransitionPane.js +107 -0
  227. package/dist/src/components/StateTransitionPane.js.map +1 -0
  228. package/dist/src/components/StatusPane.js +37 -0
  229. package/dist/src/components/StatusPane.js.map +1 -0
  230. package/dist/src/components/TaskPane.js +96 -0
  231. package/dist/src/components/TaskPane.js.map +1 -0
  232. package/dist/src/components/Welcome.js +52 -0
  233. package/dist/src/components/Welcome.js.map +1 -0
  234. package/dist/src/components/WelcomeChecklist.js +96 -0
  235. package/dist/src/components/WelcomeChecklist.js.map +1 -0
  236. package/dist/src/components/WelcomeCommands.js +61 -0
  237. package/dist/src/components/WelcomeCommands.js.map +1 -0
  238. package/dist/src/components/autocomplete-store.js +22 -0
  239. package/dist/src/components/autocomplete-store.js.map +1 -0
  240. package/dist/src/components/parse-keypress.js +174 -0
  241. package/dist/src/components/parse-keypress.js.map +1 -0
  242. package/dist/src/config.js +249 -0
  243. package/dist/src/config.js.map +1 -0
  244. package/dist/src/execution-controller.js +92 -0
  245. package/dist/src/execution-controller.js.map +1 -0
  246. package/dist/src/experience-tracker.js +294 -0
  247. package/dist/src/experience-tracker.js.map +1 -0
  248. package/dist/src/explorbot.js +348 -0
  249. package/dist/src/explorbot.js.map +1 -0
  250. package/dist/src/explorer.js +611 -0
  251. package/dist/src/explorer.js.map +1 -0
  252. package/dist/src/index.js +56 -0
  253. package/dist/src/index.js.map +1 -0
  254. package/dist/src/knowledge-tracker.js +184 -0
  255. package/dist/src/knowledge-tracker.js.map +1 -0
  256. package/dist/src/observability.js +126 -0
  257. package/dist/src/observability.js.map +1 -0
  258. package/dist/src/reporter.js +185 -0
  259. package/dist/src/reporter.js.map +1 -0
  260. package/dist/src/state-manager.js +427 -0
  261. package/dist/src/state-manager.js.map +1 -0
  262. package/dist/src/stats.js +44 -0
  263. package/dist/src/stats.js.map +1 -0
  264. package/dist/src/test-plan.js +343 -0
  265. package/dist/src/test-plan.js.map +1 -0
  266. package/dist/src/utils/aria.js +588 -0
  267. package/dist/src/utils/aria.js.map +1 -0
  268. package/dist/src/utils/code-extractor.js +21 -0
  269. package/dist/src/utils/code-extractor.js.map +1 -0
  270. package/dist/src/utils/context-formatter.js +205 -0
  271. package/dist/src/utils/context-formatter.js.map +1 -0
  272. package/dist/src/utils/error-page.js +19 -0
  273. package/dist/src/utils/error-page.js.map +1 -0
  274. package/dist/src/utils/expandable.js +35 -0
  275. package/dist/src/utils/expandable.js.map +1 -0
  276. package/dist/src/utils/hooks-runner.js +77 -0
  277. package/dist/src/utils/hooks-runner.js.map +1 -0
  278. package/dist/src/utils/html-diff.js +734 -0
  279. package/dist/src/utils/html-diff.js.map +1 -0
  280. package/dist/src/utils/html.js +1163 -0
  281. package/dist/src/utils/html.js.map +1 -0
  282. package/dist/src/utils/logger.js +465 -0
  283. package/dist/src/utils/logger.js.map +1 -0
  284. package/dist/src/utils/loop.js +126 -0
  285. package/dist/src/utils/loop.js.map +1 -0
  286. package/dist/src/utils/markdown-parser.js +117 -0
  287. package/dist/src/utils/markdown-parser.js.map +1 -0
  288. package/dist/src/utils/markdown-query.js +393 -0
  289. package/dist/src/utils/markdown-query.js.map +1 -0
  290. package/dist/src/utils/markdown-terminal.js +40 -0
  291. package/dist/src/utils/markdown-terminal.js.map +1 -0
  292. package/dist/src/utils/research-parser.js +2 -0
  293. package/dist/src/utils/research-parser.js.map +1 -0
  294. package/dist/src/utils/retry.js +55 -0
  295. package/dist/src/utils/retry.js.map +1 -0
  296. package/dist/src/utils/rules-loader.js +104 -0
  297. package/dist/src/utils/rules-loader.js.map +1 -0
  298. package/dist/src/utils/strings.js +14 -0
  299. package/dist/src/utils/strings.js.map +1 -0
  300. package/dist/src/utils/test-plan-markdown.js +301 -0
  301. package/dist/src/utils/test-plan-markdown.js.map +1 -0
  302. package/dist/src/utils/throttle.js +16 -0
  303. package/dist/src/utils/throttle.js.map +1 -0
  304. package/dist/src/utils/unique-names.js +13 -0
  305. package/dist/src/utils/unique-names.js.map +1 -0
  306. package/dist/src/utils/url-matcher.js +48 -0
  307. package/dist/src/utils/url-matcher.js.map +1 -0
  308. package/dist/src/utils/web-element.js +131 -0
  309. package/dist/src/utils/web-element.js.map +1 -0
  310. package/dist/src/utils/xpath.js +110 -0
  311. package/dist/src/utils/xpath.js.map +1 -0
  312. package/package.json +119 -0
  313. package/prompts/audit-rules.md +124 -0
  314. package/rules/chief/general.md +11 -0
  315. package/rules/chief/styles/curious.md +12 -0
  316. package/rules/chief/styles/hacker.md +19 -0
  317. package/rules/chief/styles/normal.md +11 -0
  318. package/rules/chief/styles/psycho.md +17 -0
  319. package/rules/navigator/multiple-locator.md +47 -0
  320. package/rules/navigator/output.md +69 -0
  321. package/rules/navigator/verification-actions.md +122 -0
  322. package/rules/navigator/verification-output.md +53 -0
  323. package/rules/planner/styles/curious.md +39 -0
  324. package/rules/planner/styles/normal.md +21 -0
  325. package/rules/planner/styles/psycho.md +14 -0
  326. package/rules/researcher/list-element.md +11 -0
  327. package/rules/researcher/screenshot-ui-map.md +30 -0
  328. package/rules/researcher/section-ui-map.md +18 -0
  329. package/rules/researcher/ui-map-table.md +18 -0
@@ -0,0 +1,67 @@
1
+ import { existsSync, readdirSync, rmSync, statSync, unlinkSync } from 'node:fs';
2
+ import { dirname, join } from 'node:path';
3
+ import { ConfigParser, outputPath } from '../config.js';
4
+ import { tag } from '../utils/logger.js';
5
+ import { BaseCommand } from './base-command.js';
6
+ export const CLEAN_TARGETS = {
7
+ states: { description: 'page states', getDir: () => outputPath('states') },
8
+ research: { description: 'research cache', getDir: () => outputPath('research') },
9
+ plans: { description: 'test plans', getDir: () => outputPath('plans') },
10
+ experiences: { description: 'experience files', getDir: () => getExperienceDir() },
11
+ output: { description: 'all output files', getDir: () => outputPath() },
12
+ };
13
+ function getExperienceDir() {
14
+ const configParser = ConfigParser.getInstance();
15
+ const config = configParser.getConfig();
16
+ const configPath = configParser.getConfigPath();
17
+ if (configPath) {
18
+ return join(dirname(configPath), config.dirs?.experience || 'experience');
19
+ }
20
+ return config.dirs?.experience || 'experience';
21
+ }
22
+ function cleanDirectoryContents(dirPath) {
23
+ if (!existsSync(dirPath))
24
+ return 0;
25
+ let count = 0;
26
+ for (const item of readdirSync(dirPath)) {
27
+ const itemPath = join(dirPath, item);
28
+ if (statSync(itemPath).isDirectory()) {
29
+ count += cleanDirectoryContents(itemPath);
30
+ rmSync(itemPath, { recursive: true });
31
+ }
32
+ else {
33
+ unlinkSync(itemPath);
34
+ count++;
35
+ }
36
+ }
37
+ return count;
38
+ }
39
+ export class CleanCommand extends BaseCommand {
40
+ name = 'clean';
41
+ description = 'Clean files: clean [states|research|plans|experiences|output]';
42
+ suggestions = Object.keys(CLEAN_TARGETS).map((t) => `/clean ${t}`);
43
+ async execute(args) {
44
+ const target = args.trim().toLowerCase();
45
+ if (!target) {
46
+ this.cleanTarget('output');
47
+ this.cleanTarget('experiences');
48
+ return;
49
+ }
50
+ if (!CLEAN_TARGETS[target]) {
51
+ tag('error').log(`Unknown clean target: ${target}. Available: ${Object.keys(CLEAN_TARGETS).join(', ')}`);
52
+ return;
53
+ }
54
+ this.cleanTarget(target);
55
+ }
56
+ cleanTarget(name) {
57
+ const target = CLEAN_TARGETS[name];
58
+ const dir = target.getDir();
59
+ if (!existsSync(dir)) {
60
+ tag('info').log(`${name}: nothing to clean (${dir} not found)`);
61
+ return;
62
+ }
63
+ const count = cleanDirectoryContents(dir);
64
+ tag('success').log(`Cleaned ${count} ${target.description} files from ${dir}`);
65
+ }
66
+ }
67
+ //# sourceMappingURL=clean-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"clean-command.js","sourceRoot":"","sources":["../../../src/commands/clean-command.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import { tag } from '../utils/logger.js';
2
+ import { BaseCommand } from './base-command.js';
3
+ export class ContextAriaCommand extends BaseCommand {
4
+ name = 'context:aria';
5
+ description = 'Print full ARIA snapshot for current page';
6
+ async execute(_args) {
7
+ const state = this.explorBot.getExplorer().getStateManager().getCurrentState();
8
+ if (!state) {
9
+ throw new Error('No active page to snapshot');
10
+ }
11
+ const ariaSnapshot = state.ariaSnapshot;
12
+ if (!ariaSnapshot) {
13
+ throw new Error('No ARIA snapshot available for current page');
14
+ }
15
+ tag('multiline').log(`ARIA Snapshot:\n\n${ariaSnapshot}`);
16
+ }
17
+ }
18
+ //# sourceMappingURL=context-aria-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-aria-command.js","sourceRoot":"","sources":["../../../src/commands/context-aria-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,kBAAmB,SAAQ,WAAW;IACjD,IAAI,GAAG,cAAc,CAAC;IACtB,WAAW,GAAG,2CAA2C,CAAC;IAE1D,KAAK,CAAC,OAAO,CAAC,KAAa;QACzB,MAAM,KAAK,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC,eAAe,EAAE,CAAC,eAAe,EAAE,CAAC;QAE/E,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAChD,CAAC;QAED,MAAM,YAAY,GAAG,KAAK,CAAC,YAAY,CAAC;QACxC,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;QACjE,CAAC;QAED,GAAG,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,qBAAqB,YAAY,EAAE,CAAC,CAAC;IAC5D,CAAC;CACF"}
@@ -0,0 +1,57 @@
1
+ import { Researcher } from '../ai/researcher.js';
2
+ import { outputPath } from '../config.js';
3
+ import { formatContextSummary } from '../utils/context-formatter.js';
4
+ import { tag } from '../utils/logger.js';
5
+ import { extractValidContainers } from '../utils/research-parser.js';
6
+ import { BaseCommand } from './base-command.js';
7
+ export class ContextCommand extends BaseCommand {
8
+ name = 'context';
9
+ description = 'Show page context summary (URL, headings, experience, knowledge, ARIA, HTML, research)';
10
+ suggestions = ['context:aria', 'context:html', 'context:knowledge', 'context:experience', 'context:data'];
11
+ async execute(args) {
12
+ const explorer = this.explorBot.getExplorer();
13
+ const state = explorer.getStateManager().getCurrentState();
14
+ if (!state) {
15
+ throw new Error('No active page to show context for');
16
+ }
17
+ const isVisual = args.includes('--visual') || args.includes('--screenshot');
18
+ await explorer.annotateElements();
19
+ if (isVisual) {
20
+ const cachedResearch = Researcher.getCachedResearch(state);
21
+ const containers = cachedResearch ? extractValidContainers(cachedResearch) : [];
22
+ await explorer.visuallyAnnotateElements({ containers });
23
+ }
24
+ const actionResult = await explorer.createAction().capturePageState({ includeScreenshot: isVisual });
25
+ const experienceTracker = explorer.getStateManager().getExperienceTracker();
26
+ const knowledgeTracker = this.explorBot.getKnowledgeTracker();
27
+ let mode = 'compact';
28
+ if (args.includes('--full')) {
29
+ mode = 'full';
30
+ }
31
+ else if (args.includes('--attached')) {
32
+ mode = 'attached';
33
+ }
34
+ const contextData = {
35
+ url: actionResult.url,
36
+ title: actionResult.title,
37
+ headings: {
38
+ h1: actionResult.h1,
39
+ h2: actionResult.h2,
40
+ h3: actionResult.h3,
41
+ h4: actionResult.h4,
42
+ },
43
+ experience: experienceTracker.getRelevantExperience(actionResult),
44
+ knowledge: knowledgeTracker.getRelevantKnowledge(actionResult),
45
+ ariaSnapshot: actionResult.ariaSnapshot,
46
+ combinedHtml: mode === 'full' ? await actionResult.combinedHtml() : undefined,
47
+ research: Researcher.getCachedResearch(state),
48
+ };
49
+ const output = formatContextSummary(contextData, mode);
50
+ tag('multiline').log(output);
51
+ if (isVisual && actionResult.screenshotFile) {
52
+ const fullPath = outputPath('states', actionResult.screenshotFile);
53
+ tag('info').log(`Screenshot saved: file://${fullPath}`);
54
+ }
55
+ }
56
+ }
57
+ //# sourceMappingURL=context-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-command.js","sourceRoot":"","sources":["../../../src/commands/context-command.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,25 @@
1
+ import { ActionResult } from '../action-result.js';
2
+ import { tag } from '../utils/logger.js';
3
+ import { BaseCommand } from './base-command.js';
4
+ export class ContextDataCommand extends BaseCommand {
5
+ name = 'context:data';
6
+ description = 'Extract structured data from current page';
7
+ async execute(_args) {
8
+ const explorer = this.explorBot.getExplorer();
9
+ const state = explorer.getStateManager().getCurrentState();
10
+ if (!state) {
11
+ throw new Error('No active page to extract data from');
12
+ }
13
+ const actionResult = ActionResult.fromState(state);
14
+ if (!actionResult.html || actionResult.html.trim().length < 100) {
15
+ tag('info').log('Capturing fresh page content...');
16
+ const freshResult = await explorer.createAction().capturePageState();
17
+ const table = await this.explorBot.agentResearcher().extractData(freshResult);
18
+ tag('multiline').log(table);
19
+ return;
20
+ }
21
+ const table = await this.explorBot.agentResearcher().extractData(state);
22
+ tag('multiline').log(table);
23
+ }
24
+ }
25
+ //# sourceMappingURL=context-data-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-data-command.js","sourceRoot":"","sources":["../../../src/commands/context-data-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,kBAAmB,SAAQ,WAAW;IACjD,IAAI,GAAG,cAAc,CAAC;IACtB,WAAW,GAAG,2CAA2C,CAAC;IAE1D,KAAK,CAAC,OAAO,CAAC,KAAa;QACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;QAC9C,MAAM,KAAK,GAAG,QAAQ,CAAC,eAAe,EAAE,CAAC,eAAe,EAAE,CAAC;QAE3D,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,qCAAqC,CAAC,CAAC;QACzD,CAAC;QAED,MAAM,YAAY,GAAG,YAAY,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAEnD,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,YAAY,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;YAChE,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;YACnD,MAAM,WAAW,GAAG,MAAM,QAAQ,CAAC,YAAY,EAAE,CAAC,gBAAgB,EAAE,CAAC;YACrE,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,eAAe,EAAE,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;YAC9E,GAAG,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAC5B,OAAO;QACT,CAAC;QAED,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,eAAe,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QACxE,GAAG,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;IAC9B,CAAC;CACF"}
@@ -0,0 +1,41 @@
1
+ import { basename } from 'node:path';
2
+ import chalk from 'chalk';
3
+ import { ActionResult } from '../action-result.js';
4
+ import { tag } from '../utils/logger.js';
5
+ import { BaseCommand } from './base-command.js';
6
+ export class ContextExperienceCommand extends BaseCommand {
7
+ name = 'context:experience';
8
+ description = 'Print all matching experience for current page';
9
+ async execute(_args) {
10
+ const explorer = this.explorBot.getExplorer();
11
+ const state = explorer.getStateManager().getCurrentState();
12
+ if (!state) {
13
+ throw new Error('No active page');
14
+ }
15
+ const actionResult = ActionResult.fromState(state);
16
+ const experienceTracker = explorer.getStateManager().getExperienceTracker();
17
+ const experience = experienceTracker.getRelevantExperience(actionResult);
18
+ if (experience.length === 0) {
19
+ tag('info').log(`No experience found for: ${actionResult.url}`);
20
+ return;
21
+ }
22
+ const lines = [];
23
+ lines.push(chalk.bold.cyan(`📁 Experience for ${actionResult.url} (${experience.length} files)`));
24
+ lines.push('');
25
+ for (const exp of experience) {
26
+ lines.push(chalk.yellow(`--- ${basename(exp.filePath)} ---`));
27
+ if (exp.data?.url) {
28
+ lines.push(chalk.gray(`URL: ${exp.data.url}`));
29
+ }
30
+ if (exp.data?.title) {
31
+ lines.push(chalk.gray(`Title: ${exp.data.title}`));
32
+ }
33
+ if (exp.content.trim()) {
34
+ lines.push(exp.content.trim());
35
+ }
36
+ lines.push('');
37
+ }
38
+ tag('multiline').log(lines.join('\n'));
39
+ }
40
+ }
41
+ //# sourceMappingURL=context-experience-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-experience-command.js","sourceRoot":"","sources":["../../../src/commands/context-experience-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AACrC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,wBAAyB,SAAQ,WAAW;IACvD,IAAI,GAAG,oBAAoB,CAAC;IAC5B,WAAW,GAAG,gDAAgD,CAAC;IAE/D,KAAK,CAAC,OAAO,CAAC,KAAa;QACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;QAC9C,MAAM,KAAK,GAAG,QAAQ,CAAC,eAAe,EAAE,CAAC,eAAe,EAAE,CAAC;QAE3D,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;QACpC,CAAC;QAED,MAAM,YAAY,GAAG,YAAY,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QACnD,MAAM,iBAAiB,GAAG,QAAQ,CAAC,eAAe,EAAE,CAAC,oBAAoB,EAAE,CAAC;QAC5E,MAAM,UAAU,GAAG,iBAAiB,CAAC,qBAAqB,CAAC,YAAY,CAAC,CAAC;QAEzE,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC5B,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,4BAA4B,YAAY,CAAC,GAAG,EAAE,CAAC,CAAC;YAChE,OAAO;QACT,CAAC;QAED,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,qBAAqB,YAAY,CAAC,GAAG,KAAK,UAAU,CAAC,MAAM,SAAS,CAAC,CAAC,CAAC;QAClG,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,KAAK,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;YAC7B,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,OAAO,QAAQ,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;YAC9D,IAAI,GAAG,CAAC,IAAI,EAAE,GAAG,EAAE,CAAC;gBAClB,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;YACjD,CAAC;YACD,IAAI,GAAG,CAAC,IAAI,EAAE,KAAK,EAAE,CAAC;gBACpB,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;YACrD,CAAC;YACD,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC;gBACvB,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;YACjC,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,GAAG,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IACzC,CAAC;CACF"}
@@ -0,0 +1,26 @@
1
+ import { ActionResult } from '../action-result.js';
2
+ import { tag } from '../utils/logger.js';
3
+ import { BaseCommand } from './base-command.js';
4
+ export class ContextHtmlCommand extends BaseCommand {
5
+ name = 'context:html';
6
+ description = 'Print combined HTML snapshot for current page';
7
+ async execute(_args) {
8
+ const explorer = this.explorBot.getExplorer();
9
+ const manager = explorer.getStateManager();
10
+ const state = manager.getCurrentState();
11
+ if (!state) {
12
+ throw new Error('No active page to snapshot');
13
+ }
14
+ let actionResult = ActionResult.fromState(state);
15
+ if (!actionResult.html || actionResult.html.trim().length < 100) {
16
+ tag('info').log('Capturing fresh page content...');
17
+ actionResult = await explorer.createAction().capturePageState();
18
+ }
19
+ const html = await actionResult.combinedHtml();
20
+ if (!html) {
21
+ throw new Error('No HTML snapshot available for current page');
22
+ }
23
+ tag('html').log(html);
24
+ }
25
+ }
26
+ //# sourceMappingURL=context-html-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-html-command.js","sourceRoot":"","sources":["../../../src/commands/context-html-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,kBAAmB,SAAQ,WAAW;IACjD,IAAI,GAAG,cAAc,CAAC;IACtB,WAAW,GAAG,+CAA+C,CAAC;IAE9D,KAAK,CAAC,OAAO,CAAC,KAAa;QACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;QAC9C,MAAM,OAAO,GAAG,QAAQ,CAAC,eAAe,EAAE,CAAC;QAC3C,MAAM,KAAK,GAAG,OAAO,CAAC,eAAe,EAAE,CAAC;QAExC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAChD,CAAC;QAED,IAAI,YAAY,GAAG,YAAY,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QAEjD,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,YAAY,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;YAChE,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;YACnD,YAAY,GAAG,MAAM,QAAQ,CAAC,YAAY,EAAE,CAAC,gBAAgB,EAAE,CAAC;QAClE,CAAC;QAED,MAAM,IAAI,GAAG,MAAM,YAAY,CAAC,YAAY,EAAE,CAAC;QAE/C,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,MAAM,IAAI,KAAK,CAAC,6CAA6C,CAAC,CAAC;QACjE,CAAC;QAED,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;CACF"}
@@ -0,0 +1,36 @@
1
+ import { basename } from 'node:path';
2
+ import chalk from 'chalk';
3
+ import { ActionResult } from '../action-result.js';
4
+ import { tag } from '../utils/logger.js';
5
+ import { BaseCommand } from './base-command.js';
6
+ export class ContextKnowledgeCommand extends BaseCommand {
7
+ name = 'context:knowledge';
8
+ description = 'Print all matching knowledge for current page';
9
+ async execute(_args) {
10
+ const explorer = this.explorBot.getExplorer();
11
+ const state = explorer.getStateManager().getCurrentState();
12
+ if (!state) {
13
+ throw new Error('No active page');
14
+ }
15
+ const actionResult = ActionResult.fromState(state);
16
+ const knowledgeTracker = this.explorBot.getKnowledgeTracker();
17
+ const knowledge = knowledgeTracker.getRelevantKnowledge(actionResult);
18
+ if (knowledge.length === 0) {
19
+ tag('info').log(`No knowledge found for: ${actionResult.url}`);
20
+ return;
21
+ }
22
+ const lines = [];
23
+ lines.push(chalk.bold.cyan(`📚 Knowledge for ${actionResult.url} (${knowledge.length} files)`));
24
+ lines.push('');
25
+ for (const k of knowledge) {
26
+ lines.push(chalk.yellow(`--- ${basename(k.filePath)} ---`));
27
+ lines.push(chalk.gray(`Pattern: ${k.url}`));
28
+ if (k.content.trim()) {
29
+ lines.push(k.content.trim());
30
+ }
31
+ lines.push('');
32
+ }
33
+ tag('multiline').log(lines.join('\n'));
34
+ }
35
+ }
36
+ //# sourceMappingURL=context-knowledge-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context-knowledge-command.js","sourceRoot":"","sources":["../../../src/commands/context-knowledge-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AACrC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,uBAAwB,SAAQ,WAAW;IACtD,IAAI,GAAG,mBAAmB,CAAC;IAC3B,WAAW,GAAG,+CAA+C,CAAC;IAE9D,KAAK,CAAC,OAAO,CAAC,KAAa;QACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;QAC9C,MAAM,KAAK,GAAG,QAAQ,CAAC,eAAe,EAAE,CAAC,eAAe,EAAE,CAAC;QAE3D,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;QACpC,CAAC;QAED,MAAM,YAAY,GAAG,YAAY,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QACnD,MAAM,gBAAgB,GAAG,IAAI,CAAC,SAAS,CAAC,mBAAmB,EAAE,CAAC;QAC9D,MAAM,SAAS,GAAG,gBAAgB,CAAC,oBAAoB,CAAC,YAAY,CAAC,CAAC;QAEtE,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC3B,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,2BAA2B,YAAY,CAAC,GAAG,EAAE,CAAC,CAAC;YAC/D,OAAO;QACT,CAAC;QAED,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,oBAAoB,YAAY,CAAC,GAAG,KAAK,SAAS,CAAC,MAAM,SAAS,CAAC,CAAC,CAAC;QAChG,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,KAAK,MAAM,CAAC,IAAI,SAAS,EAAE,CAAC;YAC1B,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,OAAO,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;YAC5D,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;YAC5C,IAAI,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC;gBACrB,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;YAC/B,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,GAAG,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IACzC,CAAC;CACF"}
@@ -0,0 +1,12 @@
1
+ import { isDebugMode, setDebugMode, tag } from '../utils/logger.js';
2
+ import { BaseCommand } from './base-command.js';
3
+ export class DebugCommand extends BaseCommand {
4
+ name = 'debug';
5
+ description = 'Toggle debug output';
6
+ async execute(_args) {
7
+ const enabled = !isDebugMode();
8
+ setDebugMode(enabled);
9
+ tag('info').log(`Debug mode ${enabled ? 'enabled' : 'disabled'}`);
10
+ }
11
+ }
12
+ //# sourceMappingURL=debug-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"debug-command.js","sourceRoot":"","sources":["../../../src/commands/debug-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,YAAY,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AACpE,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,YAAa,SAAQ,WAAW;IAC3C,IAAI,GAAG,OAAO,CAAC;IACf,WAAW,GAAG,qBAAqB,CAAC;IAEpC,KAAK,CAAC,OAAO,CAAC,KAAa;QACzB,MAAM,OAAO,GAAG,CAAC,WAAW,EAAE,CAAC;QAC/B,YAAY,CAAC,OAAO,CAAC,CAAC;QACtB,GAAG,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,cAAc,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC;IACpE,CAAC;CACF"}
@@ -0,0 +1,29 @@
1
+ import { BaseCommand } from './base-command.js';
2
+ export class DrillCommand extends BaseCommand {
3
+ name = 'drill';
4
+ description = 'Drill all components on current page to learn interactions';
5
+ aliases = ['bosun'];
6
+ suggestions = ['/research - to see UI map first', '/navigate <page> - to go to another page'];
7
+ async execute(args) {
8
+ const knowledgePath = this.parseKnowledgeArg(args);
9
+ const maxComponents = this.parseMaxArg(args);
10
+ const state = this.explorBot.getExplorer().getStateManager().getCurrentState();
11
+ if (!state) {
12
+ throw new Error('No active page to drill');
13
+ }
14
+ await this.explorBot.agentBosun().drill({
15
+ knowledgePath,
16
+ maxComponents,
17
+ interactive: true,
18
+ });
19
+ }
20
+ parseKnowledgeArg(args) {
21
+ const match = args.match(/--knowledge\s+(\S+)/);
22
+ return match ? match[1] : undefined;
23
+ }
24
+ parseMaxArg(args) {
25
+ const match = args.match(/--max\s+(\d+)/);
26
+ return match ? Number.parseInt(match[1], 10) : undefined;
27
+ }
28
+ }
29
+ //# sourceMappingURL=drill-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"drill-command.js","sourceRoot":"","sources":["../../../src/commands/drill-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,YAAa,SAAQ,WAAW;IAC3C,IAAI,GAAG,OAAO,CAAC;IACf,WAAW,GAAG,4DAA4D,CAAC;IAC3E,OAAO,GAAG,CAAC,OAAO,CAAC,CAAC;IACpB,WAAW,GAAG,CAAC,iCAAiC,EAAE,0CAA0C,CAAC,CAAC;IAE9F,KAAK,CAAC,OAAO,CAAC,IAAY;QACxB,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QACnD,MAAM,aAAa,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAE7C,MAAM,KAAK,GAAG,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC,eAAe,EAAE,CAAC,eAAe,EAAE,CAAC;QAC/E,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;QAC7C,CAAC;QAED,MAAM,IAAI,CAAC,SAAS,CAAC,UAAU,EAAE,CAAC,KAAK,CAAC;YACtC,aAAa;YACb,aAAa;YACb,WAAW,EAAE,IAAI;SAClB,CAAC,CAAC;IACL,CAAC;IAEO,iBAAiB,CAAC,IAAY;QACpC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC;QAChD,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IACtC,CAAC;IAEO,WAAW,CAAC,IAAY;QAC9B,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC;QAC1C,OAAO,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;IAC3D,CAAC;CACF"}
@@ -0,0 +1,26 @@
1
+ import { render } from 'ink';
2
+ import React from 'react';
3
+ import { StatusPane } from '../components/StatusPane.js';
4
+ import { Stats } from '../stats.js';
5
+ import { BaseCommand } from './base-command.js';
6
+ export class ExitCommand extends BaseCommand {
7
+ name = 'exit';
8
+ description = 'Exit the application';
9
+ aliases = ['quit'];
10
+ async execute(_args) {
11
+ await this.explorBot.getExplorer().stop();
12
+ if (Stats.hasActivity()) {
13
+ await new Promise((resolve) => {
14
+ const { unmount } = render(React.createElement(StatusPane, {
15
+ onComplete: () => {
16
+ unmount();
17
+ resolve();
18
+ },
19
+ }), { exitOnCtrlC: false, patchConsole: false });
20
+ });
21
+ }
22
+ console.log('\nGoodbye!');
23
+ process.exit(0);
24
+ }
25
+ }
26
+ //# sourceMappingURL=exit-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"exit-command.js","sourceRoot":"","sources":["../../../src/commands/exit-command.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,KAAK,CAAC;AAC7B,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,UAAU,EAAE,MAAM,6BAA6B,CAAC;AACzD,OAAO,EAAE,KAAK,EAAE,MAAM,aAAa,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAEhD,MAAM,OAAO,WAAY,SAAQ,WAAW;IAC1C,IAAI,GAAG,MAAM,CAAC;IACd,WAAW,GAAG,sBAAsB,CAAC;IACrC,OAAO,GAAG,CAAC,MAAM,CAAC,CAAC;IAEnB,KAAK,CAAC,OAAO,CAAC,KAAa;QACzB,MAAM,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC;QAE1C,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;YACxB,MAAM,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;gBAClC,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,CACxB,KAAK,CAAC,aAAa,CAAC,UAAU,EAAE;oBAC9B,UAAU,EAAE,GAAG,EAAE;wBACf,OAAO,EAAE,CAAC;wBACV,OAAO,EAAE,CAAC;oBACZ,CAAC;iBACF,CAAC,EACF,EAAE,WAAW,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,CAC5C,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAC1B,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;CACF"}
@@ -0,0 +1,124 @@
1
+ import figureSet from 'figures';
2
+ import path from 'node:path';
3
+ import { getStyles } from '../ai/planner/styles.js';
4
+ import { jsonToTable } from '../utils/markdown-parser.js';
5
+ import { tag } from '../utils/logger.js';
6
+ import { BaseCommand } from './base-command.js';
7
+ export class ExploreCommand extends BaseCommand {
8
+ name = 'explore';
9
+ description = 'Start web exploration';
10
+ options = [{ flags: '--max-tests <number>', description: 'Maximum number of tests to run' }];
11
+ suggestions = ['/navigate <page> - to go to another page', '/research - to analyze', '/plan <feature> - to plan testing'];
12
+ maxTests;
13
+ testsRun = 0;
14
+ completedPlans = [];
15
+ async execute(args) {
16
+ const maxTestsMatch = args.match(/--max-tests\s+(\d+)/);
17
+ if (maxTestsMatch) {
18
+ this.maxTests = Number.parseInt(maxTestsMatch[1], 10);
19
+ args = args.replace(/--max-tests\s+\d+/, '').trim();
20
+ }
21
+ const feature = args.trim() || undefined;
22
+ const mainUrl = this.explorBot.getExplorer().getStateManager().getCurrentState()?.url;
23
+ await this.runAllStyles(mainUrl, feature);
24
+ const mainPlan = this.explorBot.getCurrentPlan();
25
+ if (!mainPlan)
26
+ return;
27
+ this.completedPlans.push(mainPlan);
28
+ if (!this.isLimitReached()) {
29
+ const planner = this.explorBot.agentPlanner();
30
+ while (true) {
31
+ if (this.isLimitReached())
32
+ break;
33
+ const candidates = planner.collectSubPageCandidates(mainPlan, mainUrl || '/');
34
+ if (candidates.length === 0)
35
+ break;
36
+ const pick = await planner.pickNextSubPage(candidates);
37
+ if (!pick)
38
+ break;
39
+ tag('info').log(`Exploring sub-page: ${pick.url} (${pick.reason})`);
40
+ try {
41
+ await this.explorBot.visit(pick.url);
42
+ await this.runAllStyles(pick.url, undefined, mainPlan, this.completedPlans);
43
+ const subPlan = this.explorBot.getCurrentPlan();
44
+ if (subPlan) {
45
+ this.completedPlans.push(subPlan);
46
+ }
47
+ }
48
+ catch (err) {
49
+ tag('warning').log(`Sub-page exploration failed: ${err instanceof Error ? err.message : err}`);
50
+ }
51
+ }
52
+ }
53
+ this.explorBot.setCurrentPlan(mainPlan);
54
+ if (mainUrl)
55
+ await this.explorBot.visit(mainUrl);
56
+ const savedPath = this.explorBot.savePlans(this.completedPlans);
57
+ this.printResults(savedPath);
58
+ }
59
+ async runAllStyles(pageUrl, feature, parentPlan, completedPlans) {
60
+ let fresh = true;
61
+ for (const style of Object.keys(getStyles())) {
62
+ if (!fresh && pageUrl) {
63
+ await this.explorBot.visit(pageUrl);
64
+ }
65
+ const opts = { fresh, style, completedPlans };
66
+ if (fresh && parentPlan)
67
+ opts.extend = parentPlan;
68
+ await this.explorBot.plan(feature, opts);
69
+ await this.runPendingTests();
70
+ fresh = false;
71
+ }
72
+ }
73
+ printResults(savedPath) {
74
+ const allTests = this.completedPlans.flatMap((plan) => plan.tests.filter((t) => t.status !== 'pending').map((test) => ({ test, planTitle: plan.title })));
75
+ if (allTests.length === 0)
76
+ return;
77
+ const hasSubPages = this.completedPlans.length > 1;
78
+ const rows = allTests.map(({ test, planTitle }, index) => {
79
+ const durationMs = test.getDurationMs();
80
+ const duration = durationMs != null ? `${(durationMs / 1000).toFixed(1)}s` : '-';
81
+ let status = 'failed';
82
+ if (test.isSuccessful)
83
+ status = 'passed';
84
+ else if (test.isSkipped)
85
+ status = 'skipped';
86
+ const row = {
87
+ '#': String(index + 1),
88
+ Status: status,
89
+ Title: test.scenario.replace(/\|/g, '-'),
90
+ Priority: test.priority,
91
+ Time: duration,
92
+ Steps: String(Object.keys(test.notes).length),
93
+ };
94
+ if (hasSubPages) {
95
+ row.Plan = planTitle;
96
+ }
97
+ return row;
98
+ });
99
+ const columns = ['#', 'Status', 'Title', 'Priority', 'Time', 'Steps'];
100
+ if (hasSubPages)
101
+ columns.push('Plan');
102
+ tag('multiline').log(jsonToTable(rows, columns));
103
+ tag('info').log(`${figureSet.tick} ${allTests.length} tests completed`);
104
+ if (savedPath) {
105
+ const relativePath = path.relative(process.cwd(), savedPath);
106
+ tag('info').log(`Re-run tests: explorbot test ${relativePath} <index>`);
107
+ }
108
+ }
109
+ isLimitReached() {
110
+ return this.maxTests != null && this.testsRun >= this.maxTests;
111
+ }
112
+ async runPendingTests() {
113
+ const plan = this.explorBot.getCurrentPlan();
114
+ if (!plan)
115
+ return;
116
+ for (const test of plan.getPendingTests()) {
117
+ if (this.isLimitReached())
118
+ break;
119
+ await this.explorBot.agentTester().test(test);
120
+ this.testsRun++;
121
+ }
122
+ }
123
+ }
124
+ //# sourceMappingURL=explore-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"explore-command.js","sourceRoot":"","sources":["../../../src/commands/explore-command.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,84 @@
1
+ import { Planner } from '../ai/planner.js';
2
+ import { Researcher } from '../ai/researcher.js';
3
+ import { tag } from '../utils/logger.js';
4
+ import { loop } from '../utils/loop.js';
5
+ import { BaseCommand } from './base-command.js';
6
+ import { ExploreCommand } from './explore-command.js';
7
+ export class FreesailCommand extends BaseCommand {
8
+ name = 'freesail';
9
+ description = 'Continuously explore and navigate to new pages autonomously';
10
+ aliases = ['freeride'];
11
+ tuiEnabled = true;
12
+ options = [
13
+ { flags: '--deep', description: 'Use deep navigation strategy' },
14
+ { flags: '--shallow', description: 'Use shallow navigation strategy' },
15
+ { flags: '--scope <url>', description: 'Limit navigation to URLs starting with this prefix' },
16
+ { flags: '--max-tests <number>', description: 'Maximum number of tests to run' },
17
+ ];
18
+ async execute(args) {
19
+ const { strategy, scope, maxTests } = parseArgs(args);
20
+ await this.explorBot.visitInitialState();
21
+ let testsRun = 0;
22
+ await loop(async (ctx) => {
23
+ if (maxTests != null && testsRun >= maxTests)
24
+ ctx.stop();
25
+ const stateManager = this.explorBot.getExplorer().getStateManager();
26
+ const state = stateManager.getCurrentState();
27
+ if (state && !Researcher.getCachedResearch(state)) {
28
+ await this.explorBot.agentResearcher().research(state, { deep: true, screenshot: true });
29
+ }
30
+ const cachedPlan = state?.url ? Planner.getCachedPlan(state.url) : null;
31
+ if (cachedPlan?.tests.some((t) => t.result)) {
32
+ tag('info').log(`Page already tested (${cachedPlan.tests.length} tests in plan), skipping exploration`);
33
+ }
34
+ else {
35
+ const exploreCmd = new ExploreCommand(this.explorBot);
36
+ if (maxTests != null)
37
+ exploreCmd.maxTests = maxTests - testsRun;
38
+ await exploreCmd.execute('');
39
+ const plan = this.explorBot.getCurrentPlan();
40
+ if (plan)
41
+ testsRun += plan.tests.filter((t) => t.hasFinished).length;
42
+ }
43
+ if (maxTests != null && testsRun >= maxTests)
44
+ ctx.stop();
45
+ const navigator = this.explorBot.agentNavigator();
46
+ const visitedUrls = stateManager.getAllVisitedUrls();
47
+ const suggestion = await navigator.freeSail({ strategy, scope, visitedUrls });
48
+ if (!suggestion) {
49
+ tag('info').log('No navigation suggestion available');
50
+ return;
51
+ }
52
+ if (scope && !suggestion.target.startsWith(scope)) {
53
+ tag('warning').log(`Suggestion ${suggestion.target} is outside scope ${scope}, skipping`);
54
+ return;
55
+ }
56
+ tag('info').log(`Navigating to: ${suggestion.target} - ${suggestion.reason}`);
57
+ await this.explorBot.openFreshTab();
58
+ await this.explorBot.visit(suggestion.target);
59
+ this.explorBot.clearPlan();
60
+ }, { maxAttempts: Number.POSITIVE_INFINITY });
61
+ }
62
+ }
63
+ function parseArgs(args) {
64
+ const parts = args.trim().split(/\s+/);
65
+ let strategy;
66
+ let scope;
67
+ let maxTests;
68
+ for (let i = 0; i < parts.length; i++) {
69
+ if (parts[i] === '--deep')
70
+ strategy = 'deep';
71
+ if (parts[i] === '--shallow')
72
+ strategy = 'shallow';
73
+ if (parts[i] === '--scope' && parts[i + 1]) {
74
+ scope = parts[i + 1];
75
+ i++;
76
+ }
77
+ if (parts[i] === '--max-tests' && parts[i + 1]) {
78
+ maxTests = Number.parseInt(parts[i + 1], 10);
79
+ i++;
80
+ }
81
+ }
82
+ return { strategy, scope, maxTests };
83
+ }
84
+ //# sourceMappingURL=freesail-command.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"freesail-command.js","sourceRoot":"","sources":["../../../src/commands/freesail-command.ts"],"names":[],"mappings":"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"}