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.
- package/LICENSE +94 -0
- package/README.md +267 -0
- package/assets/sample-files/sample.docx +0 -0
- package/assets/sample-files/sample.mp3 +0 -0
- package/assets/sample-files/sample.mp4 +0 -0
- package/assets/sample-files/sample.pdf +21 -0
- package/assets/sample-files/sample.png +0 -0
- package/assets/sample-files/sample.xlsx +0 -0
- package/assets/sample-files/sample.zip +0 -0
- package/dist/assets/sample-files/sample.docx +0 -0
- package/dist/assets/sample-files/sample.mp3 +0 -0
- package/dist/assets/sample-files/sample.mp4 +0 -0
- package/dist/assets/sample-files/sample.pdf +21 -0
- package/dist/assets/sample-files/sample.png +0 -0
- package/dist/assets/sample-files/sample.xlsx +0 -0
- package/dist/assets/sample-files/sample.zip +0 -0
- package/dist/bin/explorbot-cli.js +683 -0
- package/dist/bin/explorbot-cli.js.map +1 -0
- package/dist/boat/api-tester/bin/apibot-cli.js +5 -0
- package/dist/boat/api-tester/bin/apibot-cli.js.map +1 -0
- package/dist/boat/api-tester/example/apibot.config.js +31 -0
- package/dist/boat/api-tester/example/apibot.config.js.map +1 -0
- package/dist/boat/api-tester/src/ai/chief/styles.js +13 -0
- package/dist/boat/api-tester/src/ai/chief/styles.js.map +1 -0
- package/dist/boat/api-tester/src/ai/chief.js +301 -0
- package/dist/boat/api-tester/src/ai/chief.js.map +1 -0
- package/dist/boat/api-tester/src/ai/curler-tools.js +263 -0
- package/dist/boat/api-tester/src/ai/curler-tools.js.map +1 -0
- package/dist/boat/api-tester/src/ai/curler.js +271 -0
- package/dist/boat/api-tester/src/ai/curler.js.map +1 -0
- package/dist/boat/api-tester/src/api-client.js +26 -0
- package/dist/boat/api-tester/src/api-client.js.map +1 -0
- package/dist/boat/api-tester/src/apibot.js +166 -0
- package/dist/boat/api-tester/src/apibot.js.map +1 -0
- package/dist/boat/api-tester/src/cli.js +262 -0
- package/dist/boat/api-tester/src/cli.js.map +1 -0
- package/dist/boat/api-tester/src/config.js +159 -0
- package/dist/boat/api-tester/src/config.js.map +1 -0
- package/dist/prompts/audit-rules.md +124 -0
- package/dist/rules/chief/general.md +11 -0
- package/dist/rules/chief/styles/curious.md +12 -0
- package/dist/rules/chief/styles/hacker.md +19 -0
- package/dist/rules/chief/styles/normal.md +11 -0
- package/dist/rules/chief/styles/psycho.md +17 -0
- package/dist/rules/navigator/multiple-locator.md +47 -0
- package/dist/rules/navigator/output.md +69 -0
- package/dist/rules/navigator/verification-actions.md +122 -0
- package/dist/rules/navigator/verification-output.md +53 -0
- package/dist/rules/planner/styles/curious.md +39 -0
- package/dist/rules/planner/styles/normal.md +21 -0
- package/dist/rules/planner/styles/psycho.md +14 -0
- package/dist/rules/researcher/list-element.md +11 -0
- package/dist/rules/researcher/screenshot-ui-map.md +30 -0
- package/dist/rules/researcher/section-ui-map.md +18 -0
- package/dist/rules/researcher/ui-map-table.md +18 -0
- package/dist/src/action-result.js +574 -0
- package/dist/src/action-result.js.map +1 -0
- package/dist/src/action.js +388 -0
- package/dist/src/action.js.map +1 -0
- package/dist/src/activity.js +86 -0
- package/dist/src/activity.js.map +1 -0
- package/dist/src/ai/agent.js +2 -0
- package/dist/src/ai/agent.js.map +1 -0
- package/dist/src/ai/bosun.js +443 -0
- package/dist/src/ai/bosun.js.map +1 -0
- package/dist/src/ai/captain/idle-mode.js +102 -0
- package/dist/src/ai/captain/idle-mode.js.map +1 -0
- package/dist/src/ai/captain/mixin.js +11 -0
- package/dist/src/ai/captain/mixin.js.map +1 -0
- package/dist/src/ai/captain/test-mode.js +251 -0
- package/dist/src/ai/captain/test-mode.js.map +1 -0
- package/dist/src/ai/captain/web-mode.js +124 -0
- package/dist/src/ai/captain/web-mode.js.map +1 -0
- package/dist/src/ai/captain.js +442 -0
- package/dist/src/ai/captain.js.map +1 -0
- package/dist/src/ai/conversation.js +176 -0
- package/dist/src/ai/conversation.js.map +1 -0
- package/dist/src/ai/experience-compactor.js +232 -0
- package/dist/src/ai/experience-compactor.js.map +1 -0
- package/dist/src/ai/fisherman-tools.js +154 -0
- package/dist/src/ai/fisherman-tools.js.map +1 -0
- package/dist/src/ai/fisherman.js +184 -0
- package/dist/src/ai/fisherman.js.map +1 -0
- package/dist/src/ai/historian.js +384 -0
- package/dist/src/ai/historian.js.map +1 -0
- package/dist/src/ai/navigator.js +493 -0
- package/dist/src/ai/navigator.js.map +1 -0
- package/dist/src/ai/pilot.js +684 -0
- package/dist/src/ai/pilot.js.map +1 -0
- package/dist/src/ai/planner/session-dedup.js +28 -0
- package/dist/src/ai/planner/session-dedup.js.map +1 -0
- package/dist/src/ai/planner/styles.js +15 -0
- package/dist/src/ai/planner/styles.js.map +1 -0
- package/dist/src/ai/planner/subpages.js +118 -0
- package/dist/src/ai/planner/subpages.js.map +1 -0
- package/dist/src/ai/planner.js +486 -0
- package/dist/src/ai/planner.js.map +1 -0
- package/dist/src/ai/provider.js +540 -0
- package/dist/src/ai/provider.js.map +1 -0
- package/dist/src/ai/quartermaster.js +210 -0
- package/dist/src/ai/quartermaster.js.map +1 -0
- package/dist/src/ai/researcher/cache.js +95 -0
- package/dist/src/ai/researcher/cache.js.map +1 -0
- package/dist/src/ai/researcher/coordinates.js +210 -0
- package/dist/src/ai/researcher/coordinates.js.map +1 -0
- package/dist/src/ai/researcher/deep-analysis.js +364 -0
- package/dist/src/ai/researcher/deep-analysis.js.map +1 -0
- package/dist/src/ai/researcher/fingerprint-worker.js +46 -0
- package/dist/src/ai/researcher/fingerprint-worker.js.map +1 -0
- package/dist/src/ai/researcher/focus.js +37 -0
- package/dist/src/ai/researcher/focus.js.map +1 -0
- package/dist/src/ai/researcher/locators.js +242 -0
- package/dist/src/ai/researcher/locators.js.map +1 -0
- package/dist/src/ai/researcher/mixin.js +3 -0
- package/dist/src/ai/researcher/mixin.js.map +1 -0
- package/dist/src/ai/researcher/parser.js +160 -0
- package/dist/src/ai/researcher/parser.js.map +1 -0
- package/dist/src/ai/researcher/research-result.js +110 -0
- package/dist/src/ai/researcher/research-result.js.map +1 -0
- package/dist/src/ai/researcher.js +776 -0
- package/dist/src/ai/researcher.js.map +1 -0
- package/dist/src/ai/rules.js +368 -0
- package/dist/src/ai/rules.js.map +1 -0
- package/dist/src/ai/task-agent.js +110 -0
- package/dist/src/ai/task-agent.js.map +1 -0
- package/dist/src/ai/tester.js +840 -0
- package/dist/src/ai/tester.js.map +1 -0
- package/dist/src/ai/tools.js +980 -0
- package/dist/src/ai/tools.js.map +1 -0
- package/dist/src/api/api-client.js +91 -0
- package/dist/src/api/api-client.js.map +1 -0
- package/dist/src/api/request-result.js +177 -0
- package/dist/src/api/request-result.js.map +1 -0
- package/dist/src/api/request-store.js +109 -0
- package/dist/src/api/request-store.js.map +1 -0
- package/dist/src/api/spec-reader.js +148 -0
- package/dist/src/api/spec-reader.js.map +1 -0
- package/dist/src/api/xhr-capture.js +91 -0
- package/dist/src/api/xhr-capture.js.map +1 -0
- package/dist/src/browser-server.js +67 -0
- package/dist/src/browser-server.js.map +1 -0
- package/dist/src/command-handler.js +363 -0
- package/dist/src/command-handler.js.map +1 -0
- package/dist/src/commands/add-rule-command.js +52 -0
- package/dist/src/commands/add-rule-command.js.map +1 -0
- package/dist/src/commands/base-command.js +14 -0
- package/dist/src/commands/base-command.js.map +1 -0
- package/dist/src/commands/clean-command.js +67 -0
- package/dist/src/commands/clean-command.js.map +1 -0
- package/dist/src/commands/context-aria-command.js +18 -0
- package/dist/src/commands/context-aria-command.js.map +1 -0
- package/dist/src/commands/context-command.js +57 -0
- package/dist/src/commands/context-command.js.map +1 -0
- package/dist/src/commands/context-data-command.js +25 -0
- package/dist/src/commands/context-data-command.js.map +1 -0
- package/dist/src/commands/context-experience-command.js +41 -0
- package/dist/src/commands/context-experience-command.js.map +1 -0
- package/dist/src/commands/context-html-command.js +26 -0
- package/dist/src/commands/context-html-command.js.map +1 -0
- package/dist/src/commands/context-knowledge-command.js +36 -0
- package/dist/src/commands/context-knowledge-command.js.map +1 -0
- package/dist/src/commands/debug-command.js +12 -0
- package/dist/src/commands/debug-command.js.map +1 -0
- package/dist/src/commands/drill-command.js +29 -0
- package/dist/src/commands/drill-command.js.map +1 -0
- package/dist/src/commands/exit-command.js +26 -0
- package/dist/src/commands/exit-command.js.map +1 -0
- package/dist/src/commands/explore-command.js +124 -0
- package/dist/src/commands/explore-command.js.map +1 -0
- package/dist/src/commands/freesail-command.js +84 -0
- package/dist/src/commands/freesail-command.js.map +1 -0
- package/dist/src/commands/help-command.js +7 -0
- package/dist/src/commands/help-command.js.map +1 -0
- package/dist/src/commands/index.js +63 -0
- package/dist/src/commands/index.js.map +1 -0
- package/dist/src/commands/knows-command.js +54 -0
- package/dist/src/commands/knows-command.js.map +1 -0
- package/dist/src/commands/learn-command.js +35 -0
- package/dist/src/commands/learn-command.js.map +1 -0
- package/dist/src/commands/navigate-command.js +16 -0
- package/dist/src/commands/navigate-command.js.map +1 -0
- package/dist/src/commands/path-command.js +70 -0
- package/dist/src/commands/path-command.js.map +1 -0
- package/dist/src/commands/plan-clear-command.js +13 -0
- package/dist/src/commands/plan-clear-command.js.map +1 -0
- package/dist/src/commands/plan-command.js +36 -0
- package/dist/src/commands/plan-command.js.map +1 -0
- package/dist/src/commands/plan-edit-command.js +8 -0
- package/dist/src/commands/plan-edit-command.js.map +1 -0
- package/dist/src/commands/plan-load-command.js +16 -0
- package/dist/src/commands/plan-load-command.js.map +1 -0
- package/dist/src/commands/plan-reload-command.js +23 -0
- package/dist/src/commands/plan-reload-command.js.map +1 -0
- package/dist/src/commands/plan-save-command.js +22 -0
- package/dist/src/commands/plan-save-command.js.map +1 -0
- package/dist/src/commands/research-command.js +38 -0
- package/dist/src/commands/research-command.js.map +1 -0
- package/dist/src/commands/start-command.js +12 -0
- package/dist/src/commands/start-command.js.map +1 -0
- package/dist/src/commands/status-command.js +19 -0
- package/dist/src/commands/status-command.js.map +1 -0
- package/dist/src/commands/test-command.js +85 -0
- package/dist/src/commands/test-command.js.map +1 -0
- package/dist/src/components/ActivityPane.js +55 -0
- package/dist/src/components/ActivityPane.js.map +1 -0
- package/dist/src/components/AddKnowledge.js +122 -0
- package/dist/src/components/AddKnowledge.js.map +1 -0
- package/dist/src/components/AddRule.js +117 -0
- package/dist/src/components/AddRule.js.map +1 -0
- package/dist/src/components/App.js +313 -0
- package/dist/src/components/App.js.map +1 -0
- package/dist/src/components/Autocomplete.js +43 -0
- package/dist/src/components/Autocomplete.js.map +1 -0
- package/dist/src/components/InputPane.js +207 -0
- package/dist/src/components/InputPane.js.map +1 -0
- package/dist/src/components/InputReadline.js +598 -0
- package/dist/src/components/InputReadline.js.map +1 -0
- package/dist/src/components/LogPane.js +123 -0
- package/dist/src/components/LogPane.js.map +1 -0
- package/dist/src/components/PlanEditor.js +126 -0
- package/dist/src/components/PlanEditor.js.map +1 -0
- package/dist/src/components/PlanPane.js +51 -0
- package/dist/src/components/PlanPane.js.map +1 -0
- package/dist/src/components/SessionTimer.js +26 -0
- package/dist/src/components/SessionTimer.js.map +1 -0
- package/dist/src/components/StateTransitionPane.js +107 -0
- package/dist/src/components/StateTransitionPane.js.map +1 -0
- package/dist/src/components/StatusPane.js +37 -0
- package/dist/src/components/StatusPane.js.map +1 -0
- package/dist/src/components/TaskPane.js +96 -0
- package/dist/src/components/TaskPane.js.map +1 -0
- package/dist/src/components/Welcome.js +52 -0
- package/dist/src/components/Welcome.js.map +1 -0
- package/dist/src/components/WelcomeChecklist.js +96 -0
- package/dist/src/components/WelcomeChecklist.js.map +1 -0
- package/dist/src/components/WelcomeCommands.js +61 -0
- package/dist/src/components/WelcomeCommands.js.map +1 -0
- package/dist/src/components/autocomplete-store.js +22 -0
- package/dist/src/components/autocomplete-store.js.map +1 -0
- package/dist/src/components/parse-keypress.js +174 -0
- package/dist/src/components/parse-keypress.js.map +1 -0
- package/dist/src/config.js +249 -0
- package/dist/src/config.js.map +1 -0
- package/dist/src/execution-controller.js +92 -0
- package/dist/src/execution-controller.js.map +1 -0
- package/dist/src/experience-tracker.js +294 -0
- package/dist/src/experience-tracker.js.map +1 -0
- package/dist/src/explorbot.js +348 -0
- package/dist/src/explorbot.js.map +1 -0
- package/dist/src/explorer.js +611 -0
- package/dist/src/explorer.js.map +1 -0
- package/dist/src/index.js +56 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/knowledge-tracker.js +184 -0
- package/dist/src/knowledge-tracker.js.map +1 -0
- package/dist/src/observability.js +126 -0
- package/dist/src/observability.js.map +1 -0
- package/dist/src/reporter.js +185 -0
- package/dist/src/reporter.js.map +1 -0
- package/dist/src/state-manager.js +427 -0
- package/dist/src/state-manager.js.map +1 -0
- package/dist/src/stats.js +44 -0
- package/dist/src/stats.js.map +1 -0
- package/dist/src/test-plan.js +343 -0
- package/dist/src/test-plan.js.map +1 -0
- package/dist/src/utils/aria.js +588 -0
- package/dist/src/utils/aria.js.map +1 -0
- package/dist/src/utils/code-extractor.js +21 -0
- package/dist/src/utils/code-extractor.js.map +1 -0
- package/dist/src/utils/context-formatter.js +205 -0
- package/dist/src/utils/context-formatter.js.map +1 -0
- package/dist/src/utils/error-page.js +19 -0
- package/dist/src/utils/error-page.js.map +1 -0
- package/dist/src/utils/expandable.js +35 -0
- package/dist/src/utils/expandable.js.map +1 -0
- package/dist/src/utils/hooks-runner.js +77 -0
- package/dist/src/utils/hooks-runner.js.map +1 -0
- package/dist/src/utils/html-diff.js +734 -0
- package/dist/src/utils/html-diff.js.map +1 -0
- package/dist/src/utils/html.js +1163 -0
- package/dist/src/utils/html.js.map +1 -0
- package/dist/src/utils/logger.js +465 -0
- package/dist/src/utils/logger.js.map +1 -0
- package/dist/src/utils/loop.js +126 -0
- package/dist/src/utils/loop.js.map +1 -0
- package/dist/src/utils/markdown-parser.js +117 -0
- package/dist/src/utils/markdown-parser.js.map +1 -0
- package/dist/src/utils/markdown-query.js +393 -0
- package/dist/src/utils/markdown-query.js.map +1 -0
- package/dist/src/utils/markdown-terminal.js +40 -0
- package/dist/src/utils/markdown-terminal.js.map +1 -0
- package/dist/src/utils/research-parser.js +2 -0
- package/dist/src/utils/research-parser.js.map +1 -0
- package/dist/src/utils/retry.js +55 -0
- package/dist/src/utils/retry.js.map +1 -0
- package/dist/src/utils/rules-loader.js +104 -0
- package/dist/src/utils/rules-loader.js.map +1 -0
- package/dist/src/utils/strings.js +14 -0
- package/dist/src/utils/strings.js.map +1 -0
- package/dist/src/utils/test-plan-markdown.js +301 -0
- package/dist/src/utils/test-plan-markdown.js.map +1 -0
- package/dist/src/utils/throttle.js +16 -0
- package/dist/src/utils/throttle.js.map +1 -0
- package/dist/src/utils/unique-names.js +13 -0
- package/dist/src/utils/unique-names.js.map +1 -0
- package/dist/src/utils/url-matcher.js +48 -0
- package/dist/src/utils/url-matcher.js.map +1 -0
- package/dist/src/utils/web-element.js +131 -0
- package/dist/src/utils/web-element.js.map +1 -0
- package/dist/src/utils/xpath.js +110 -0
- package/dist/src/utils/xpath.js.map +1 -0
- package/package.json +119 -0
- package/prompts/audit-rules.md +124 -0
- package/rules/chief/general.md +11 -0
- package/rules/chief/styles/curious.md +12 -0
- package/rules/chief/styles/hacker.md +19 -0
- package/rules/chief/styles/normal.md +11 -0
- package/rules/chief/styles/psycho.md +17 -0
- package/rules/navigator/multiple-locator.md +47 -0
- package/rules/navigator/output.md +69 -0
- package/rules/navigator/verification-actions.md +122 -0
- package/rules/navigator/verification-output.md +53 -0
- package/rules/planner/styles/curious.md +39 -0
- package/rules/planner/styles/normal.md +21 -0
- package/rules/planner/styles/psycho.md +14 -0
- package/rules/researcher/list-element.md +11 -0
- package/rules/researcher/screenshot-ui-map.md +30 -0
- package/rules/researcher/section-ui-map.md +18 -0
- package/rules/researcher/ui-map-table.md +18 -0
|
@@ -0,0 +1,443 @@
|
|
|
1
|
+
import { tool } from 'ai';
|
|
2
|
+
import dedent from 'dedent';
|
|
3
|
+
import { z } from 'zod';
|
|
4
|
+
import { ActionResult } from "../action-result.js";
|
|
5
|
+
import { setActivity } from "../activity.js";
|
|
6
|
+
import { Observability } from "../observability.js";
|
|
7
|
+
import { Plan, Task, Test, TestResult } from "../test-plan.js";
|
|
8
|
+
import { HooksRunner } from "../utils/hooks-runner.js";
|
|
9
|
+
import { createDebug, tag } from "../utils/logger.js";
|
|
10
|
+
import { loop, pause } from "../utils/loop.js";
|
|
11
|
+
import { locatorRule } from "./rules.js";
|
|
12
|
+
import { TaskAgent, isInteractive } from "./task-agent.js";
|
|
13
|
+
import { createCodeceptJSTools } from "./tools.js";
|
|
14
|
+
const debugLog = createDebug('explorbot:bosun');
|
|
15
|
+
export class Bosun extends TaskAgent {
|
|
16
|
+
ACTION_TOOLS = ['click', 'pressKey', 'form'];
|
|
17
|
+
emoji = '⚓';
|
|
18
|
+
explorer;
|
|
19
|
+
provider;
|
|
20
|
+
researcher;
|
|
21
|
+
navigator;
|
|
22
|
+
hooksRunner;
|
|
23
|
+
currentPlan;
|
|
24
|
+
currentConversation = null;
|
|
25
|
+
allResults = [];
|
|
26
|
+
agentTools;
|
|
27
|
+
MAX_ITERATIONS = 50;
|
|
28
|
+
constructor(explorer, provider, researcher, navigator, agentTools) {
|
|
29
|
+
super();
|
|
30
|
+
this.explorer = explorer;
|
|
31
|
+
this.provider = provider;
|
|
32
|
+
this.researcher = researcher;
|
|
33
|
+
this.navigator = navigator;
|
|
34
|
+
this.hooksRunner = new HooksRunner(explorer, explorer.getConfig());
|
|
35
|
+
this.agentTools = agentTools;
|
|
36
|
+
}
|
|
37
|
+
getNavigator() {
|
|
38
|
+
return this.navigator;
|
|
39
|
+
}
|
|
40
|
+
getExperienceTracker() {
|
|
41
|
+
return this.explorer.getStateManager().getExperienceTracker();
|
|
42
|
+
}
|
|
43
|
+
getKnowledgeTracker() {
|
|
44
|
+
return this.explorer.getKnowledgeTracker();
|
|
45
|
+
}
|
|
46
|
+
getProvider() {
|
|
47
|
+
return this.provider;
|
|
48
|
+
}
|
|
49
|
+
getSystemMessage() {
|
|
50
|
+
const currentUrl = this.explorer.getStateManager().getCurrentState()?.url;
|
|
51
|
+
const customPrompt = this.provider.getSystemPromptForAgent('bosun', currentUrl);
|
|
52
|
+
return dedent `
|
|
53
|
+
<role>
|
|
54
|
+
You are a senior QA automation engineer focused on learning how to interact with UI components.
|
|
55
|
+
Your goal is to systematically discover all possible interactions with each component and document what works.
|
|
56
|
+
</role>
|
|
57
|
+
|
|
58
|
+
<approach>
|
|
59
|
+
1. Review the UI map to understand all available components
|
|
60
|
+
2. Create a plan listing all components to drill using drill_plan tool
|
|
61
|
+
3. For each component, try appropriate interactions using click, form tools
|
|
62
|
+
4. Use drill_record to document successful interactions
|
|
63
|
+
5. If an interaction fails multiple times, use drill_ask for help (in interactive mode)
|
|
64
|
+
6. Call drill_finish when all components have been tested
|
|
65
|
+
</approach>
|
|
66
|
+
|
|
67
|
+
<rules>
|
|
68
|
+
- Focus on one component at a time
|
|
69
|
+
- Try multiple locator strategies if one fails
|
|
70
|
+
- Document what each interaction does (opens modal, navigates, etc.)
|
|
71
|
+
- Skip decorative or non-interactive elements
|
|
72
|
+
- Restore page state after each interaction (press Escape or navigate back)
|
|
73
|
+
</rules>
|
|
74
|
+
|
|
75
|
+
${locatorRule}
|
|
76
|
+
|
|
77
|
+
${customPrompt || ''}
|
|
78
|
+
`;
|
|
79
|
+
}
|
|
80
|
+
async drill(opts = {}) {
|
|
81
|
+
const { knowledgePath, maxComponents = 20, interactive = isInteractive() } = opts;
|
|
82
|
+
const state = this.explorer.getStateManager().getCurrentState();
|
|
83
|
+
if (!state)
|
|
84
|
+
throw new Error('No page state available');
|
|
85
|
+
const sessionName = `bosun_${Date.now().toString(36)}`;
|
|
86
|
+
this.allResults = [];
|
|
87
|
+
return Observability.run(`bosun: ${state.url}`, { tags: ['bosun'], sessionId: sessionName }, async () => {
|
|
88
|
+
tag('info').log(`Bosun starting drill on ${state.url}`);
|
|
89
|
+
setActivity(`${this.emoji} Researching page for drilling...`, 'action');
|
|
90
|
+
await this.hooksRunner.runBeforeHook('bosun', state.url);
|
|
91
|
+
const research = await this.researcher.research(state, { screenshot: true, force: true });
|
|
92
|
+
this.currentPlan = new Plan(`Drill: ${state.url}`);
|
|
93
|
+
this.currentPlan.url = state.url;
|
|
94
|
+
const conversation = this.provider.startConversation(this.getSystemMessage(), 'bosun');
|
|
95
|
+
this.currentConversation = conversation;
|
|
96
|
+
const initialPrompt = await this.buildInitialPrompt(state, research, maxComponents);
|
|
97
|
+
conversation.addUserText(initialPrompt);
|
|
98
|
+
const drillTask = new Task(`Drill session: ${state.url}`, state.url);
|
|
99
|
+
const codeceptjsTools = createCodeceptJSTools(this.explorer, drillTask);
|
|
100
|
+
const drillFlowTools = this.createDrillFlowTools(state, interactive);
|
|
101
|
+
const tools = {
|
|
102
|
+
...codeceptjsTools,
|
|
103
|
+
...drillFlowTools,
|
|
104
|
+
...this.agentTools,
|
|
105
|
+
};
|
|
106
|
+
let drillFinished = false;
|
|
107
|
+
await loop(async ({ stop, iteration }) => {
|
|
108
|
+
debugLog(`Drill iteration ${iteration}`);
|
|
109
|
+
setActivity(`${this.emoji} Drilling components...`, 'action');
|
|
110
|
+
const currentState = ActionResult.fromState(this.explorer.getStateManager().getCurrentState());
|
|
111
|
+
if (iteration > 1) {
|
|
112
|
+
conversation.cleanupTag('page_aria', '...cleaned aria snapshot...', 2);
|
|
113
|
+
const contextUpdate = await this.buildContextUpdate(currentState);
|
|
114
|
+
conversation.addUserText(contextUpdate);
|
|
115
|
+
}
|
|
116
|
+
const result = await this.provider.invokeConversation(conversation, tools, {
|
|
117
|
+
maxToolRoundtrips: 5,
|
|
118
|
+
toolChoice: 'required',
|
|
119
|
+
});
|
|
120
|
+
if (!result)
|
|
121
|
+
throw new Error('Failed to get response from provider');
|
|
122
|
+
const toolExecutions = result.toolExecutions || [];
|
|
123
|
+
this.trackToolExecutions(toolExecutions);
|
|
124
|
+
for (const execution of toolExecutions) {
|
|
125
|
+
if (execution.wasSuccessful && this.ACTION_TOOLS.includes(execution.toolName)) {
|
|
126
|
+
const componentName = execution.input?.explanation || 'unknown';
|
|
127
|
+
this.allResults.push({
|
|
128
|
+
component: componentName,
|
|
129
|
+
action: execution.toolName,
|
|
130
|
+
result: 'success',
|
|
131
|
+
description: execution.output?.message || 'Action completed',
|
|
132
|
+
code: execution.output?.code,
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
const finishExecution = toolExecutions.find((e) => e.toolName === 'drill_finish');
|
|
137
|
+
if (finishExecution) {
|
|
138
|
+
drillFinished = true;
|
|
139
|
+
stop();
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
if (iteration >= this.MAX_ITERATIONS) {
|
|
143
|
+
tag('warning').log('Max iterations reached');
|
|
144
|
+
stop();
|
|
145
|
+
}
|
|
146
|
+
}, {
|
|
147
|
+
maxAttempts: this.MAX_ITERATIONS,
|
|
148
|
+
interruptPrompt: 'Drill interrupted. Enter instruction (or "stop" to end):',
|
|
149
|
+
observability: {
|
|
150
|
+
agent: 'bosun',
|
|
151
|
+
sessionId: sessionName,
|
|
152
|
+
},
|
|
153
|
+
catch: async ({ error, stop }) => {
|
|
154
|
+
tag('error').log(`Drill error: ${error}`);
|
|
155
|
+
stop();
|
|
156
|
+
},
|
|
157
|
+
});
|
|
158
|
+
await this.saveToExperience(state, this.allResults);
|
|
159
|
+
if (knowledgePath) {
|
|
160
|
+
await this.saveToKnowledge(knowledgePath, state, this.allResults);
|
|
161
|
+
}
|
|
162
|
+
await this.hooksRunner.runAfterHook('bosun', state.url);
|
|
163
|
+
this.logSummary();
|
|
164
|
+
return this.currentPlan;
|
|
165
|
+
});
|
|
166
|
+
}
|
|
167
|
+
async buildInitialPrompt(state, research, maxComponents) {
|
|
168
|
+
const actionResult = ActionResult.fromState(state);
|
|
169
|
+
const knowledge = this.getKnowledge(actionResult);
|
|
170
|
+
const experience = this.getExperience(actionResult);
|
|
171
|
+
return dedent `
|
|
172
|
+
<task>
|
|
173
|
+
Drill all interactive components on this page to learn how to interact with them.
|
|
174
|
+
Maximum components to drill: ${maxComponents}
|
|
175
|
+
</task>
|
|
176
|
+
|
|
177
|
+
<page>
|
|
178
|
+
URL: ${state.url}
|
|
179
|
+
Title: ${state.title || 'Unknown'}
|
|
180
|
+
</page>
|
|
181
|
+
|
|
182
|
+
<page_ui_map>
|
|
183
|
+
${research}
|
|
184
|
+
</page_ui_map>
|
|
185
|
+
|
|
186
|
+
<page_aria>
|
|
187
|
+
${actionResult.getInteractiveARIA()}
|
|
188
|
+
</page_aria>
|
|
189
|
+
|
|
190
|
+
${knowledge}
|
|
191
|
+
${experience}
|
|
192
|
+
|
|
193
|
+
<instructions>
|
|
194
|
+
1. First, call drill_plan to create a list of components to test
|
|
195
|
+
2. Then systematically test each component using click or form tools
|
|
196
|
+
3. Use drill_record to save observations about what each component does
|
|
197
|
+
4. Press Escape or use drill_restore to reset state between tests
|
|
198
|
+
5. Call drill_finish when all components have been tested
|
|
199
|
+
</instructions>
|
|
200
|
+
`;
|
|
201
|
+
}
|
|
202
|
+
async buildContextUpdate(currentState) {
|
|
203
|
+
const remainingComponents = this.currentPlan?.tests.filter((t) => !t.hasFinished).length || 0;
|
|
204
|
+
return dedent `
|
|
205
|
+
<context_update>
|
|
206
|
+
Current URL: ${currentState.url}
|
|
207
|
+
Components remaining: ${remainingComponents}
|
|
208
|
+
Successful interactions so far: ${this.allResults.filter((r) => r.result === 'success').length}
|
|
209
|
+
</context_update>
|
|
210
|
+
|
|
211
|
+
<page_aria>
|
|
212
|
+
${currentState.getInteractiveARIA()}
|
|
213
|
+
</page_aria>
|
|
214
|
+
|
|
215
|
+
Continue drilling components. Test each one and record what it does.
|
|
216
|
+
`;
|
|
217
|
+
}
|
|
218
|
+
createDrillFlowTools(originalState, interactive) {
|
|
219
|
+
const originalUrl = originalState.url;
|
|
220
|
+
return {
|
|
221
|
+
drill_plan: tool({
|
|
222
|
+
description: 'Create a plan of components to drill. Call this first to identify all testable components from the UI map.',
|
|
223
|
+
inputSchema: z.object({
|
|
224
|
+
components: z.array(z.object({
|
|
225
|
+
name: z.string().describe('Display name of the component'),
|
|
226
|
+
role: z.string().describe('ARIA role (button, link, textbox, combobox, etc.)'),
|
|
227
|
+
locator: z.string().describe('Best locator for this component'),
|
|
228
|
+
section: z.string().optional().describe('Section of the page where component is located'),
|
|
229
|
+
})),
|
|
230
|
+
}),
|
|
231
|
+
execute: async ({ components }) => {
|
|
232
|
+
for (const comp of components) {
|
|
233
|
+
const task = new Test(`Learn: ${comp.name} (${comp.role})`, 'normal', [`Discover interactions for ${comp.name}`], originalUrl);
|
|
234
|
+
task.component = comp;
|
|
235
|
+
task.interactions = [];
|
|
236
|
+
this.currentPlan.addTest(task);
|
|
237
|
+
}
|
|
238
|
+
tag('info').log(`Created drill plan with ${components.length} components`);
|
|
239
|
+
return {
|
|
240
|
+
success: true,
|
|
241
|
+
message: `Plan created with ${components.length} components`,
|
|
242
|
+
components: components.map((c) => `${c.name} (${c.role})`),
|
|
243
|
+
instruction: 'Now test each component using click or form tools. Record observations with drill_record.',
|
|
244
|
+
};
|
|
245
|
+
},
|
|
246
|
+
}),
|
|
247
|
+
drill_record: tool({
|
|
248
|
+
description: 'Record what a component does after testing it. Call this after each successful interaction.',
|
|
249
|
+
inputSchema: z.object({
|
|
250
|
+
component: z.string().describe('Component name that was tested'),
|
|
251
|
+
action: z.string().describe('Action performed (click, form)'),
|
|
252
|
+
result: z.string().describe('What happened (opened modal, navigated to X, showed dropdown, etc.)'),
|
|
253
|
+
code: z.string().optional().describe('The CodeceptJS code that worked'),
|
|
254
|
+
}),
|
|
255
|
+
execute: async ({ component, action, result, code }) => {
|
|
256
|
+
const task = this.findComponentTask(component);
|
|
257
|
+
if (task) {
|
|
258
|
+
task.addNote(`${action}: ${result}`, TestResult.PASSED);
|
|
259
|
+
task.finish(TestResult.PASSED);
|
|
260
|
+
}
|
|
261
|
+
this.allResults.push({
|
|
262
|
+
component,
|
|
263
|
+
action,
|
|
264
|
+
result: 'success',
|
|
265
|
+
description: result,
|
|
266
|
+
code,
|
|
267
|
+
});
|
|
268
|
+
tag('success').log(`${component}: ${action} -> ${result}`);
|
|
269
|
+
return {
|
|
270
|
+
success: true,
|
|
271
|
+
recorded: `${component}: ${action} -> ${result}`,
|
|
272
|
+
instruction: 'Continue testing other components or call drill_finish when done.',
|
|
273
|
+
};
|
|
274
|
+
},
|
|
275
|
+
}),
|
|
276
|
+
drill_restore: tool({
|
|
277
|
+
description: 'Restore page state after testing a component. Use when page navigated away or modal opened.',
|
|
278
|
+
inputSchema: z.object({
|
|
279
|
+
reason: z.string().describe('Why restoration is needed'),
|
|
280
|
+
}),
|
|
281
|
+
execute: async ({ reason }) => {
|
|
282
|
+
const currentState = this.explorer.getStateManager().getCurrentState();
|
|
283
|
+
const action = this.explorer.createAction();
|
|
284
|
+
if (currentState?.url !== originalUrl) {
|
|
285
|
+
await action.execute(`I.amOnPage("${originalUrl}")`);
|
|
286
|
+
return { success: true, action: 'navigated back', url: originalUrl };
|
|
287
|
+
}
|
|
288
|
+
await action.execute('I.pressKey("Escape")');
|
|
289
|
+
return { success: true, action: 'pressed Escape' };
|
|
290
|
+
},
|
|
291
|
+
}),
|
|
292
|
+
drill_skip: tool({
|
|
293
|
+
description: 'Skip a component that cannot be drilled.',
|
|
294
|
+
inputSchema: z.object({
|
|
295
|
+
component: z.string().describe('Component to skip'),
|
|
296
|
+
reason: z.string().describe('Why this component is being skipped'),
|
|
297
|
+
}),
|
|
298
|
+
execute: async ({ component, reason }) => {
|
|
299
|
+
const task = this.findComponentTask(component);
|
|
300
|
+
if (task) {
|
|
301
|
+
task.addNote(`Skipped: ${reason}`, TestResult.FAILED);
|
|
302
|
+
task.finish(TestResult.FAILED);
|
|
303
|
+
}
|
|
304
|
+
this.allResults.push({
|
|
305
|
+
component,
|
|
306
|
+
action: 'skip',
|
|
307
|
+
result: 'unknown',
|
|
308
|
+
description: reason,
|
|
309
|
+
});
|
|
310
|
+
tag('warning').log(`Skipped ${component}: ${reason}`);
|
|
311
|
+
return { success: true, skipped: component, reason };
|
|
312
|
+
},
|
|
313
|
+
}),
|
|
314
|
+
drill_ask: tool({
|
|
315
|
+
description: 'Ask the user for help when stuck on a component. Only available in interactive mode.',
|
|
316
|
+
inputSchema: z.object({
|
|
317
|
+
component: z.string().describe('Component you need help with'),
|
|
318
|
+
question: z.string().describe('What you need help with'),
|
|
319
|
+
triedLocators: z.array(z.string()).optional().describe('Locators already tried'),
|
|
320
|
+
}),
|
|
321
|
+
execute: async ({ component, question, triedLocators }) => {
|
|
322
|
+
if (!interactive) {
|
|
323
|
+
return { success: false, message: 'Not in interactive mode. Skip this component.' };
|
|
324
|
+
}
|
|
325
|
+
let prompt = `Help needed for "${component}"\n${question}`;
|
|
326
|
+
if (triedLocators?.length) {
|
|
327
|
+
prompt += `\n\nAlready tried:\n${triedLocators.map((l) => ` - ${l}`).join('\n')}`;
|
|
328
|
+
}
|
|
329
|
+
prompt += '\n\nYour CodeceptJS command ("skip" to continue):';
|
|
330
|
+
const userInput = await pause(prompt);
|
|
331
|
+
if (!userInput || userInput.toLowerCase() === 'skip') {
|
|
332
|
+
return { success: false, skipped: true, instruction: 'Use drill_skip to skip this component.' };
|
|
333
|
+
}
|
|
334
|
+
return {
|
|
335
|
+
success: true,
|
|
336
|
+
userSuggestion: userInput,
|
|
337
|
+
instruction: `Try this command: ${userInput}`,
|
|
338
|
+
};
|
|
339
|
+
},
|
|
340
|
+
}),
|
|
341
|
+
drill_finish: tool({
|
|
342
|
+
description: 'Finish the drill session. Call when all components have been tested.',
|
|
343
|
+
inputSchema: z.object({
|
|
344
|
+
summary: z.string().describe('Summary of what was learned during drilling'),
|
|
345
|
+
}),
|
|
346
|
+
execute: async ({ summary }) => {
|
|
347
|
+
for (const test of this.currentPlan.tests) {
|
|
348
|
+
if (!test.hasFinished) {
|
|
349
|
+
test.addNote('Not tested');
|
|
350
|
+
test.finish(TestResult.FAILED);
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
tag('info').log(`Drill completed: ${summary}`);
|
|
354
|
+
return {
|
|
355
|
+
success: true,
|
|
356
|
+
totalComponents: this.currentPlan.tests.length,
|
|
357
|
+
successfulInteractions: this.allResults.filter((r) => r.result === 'success').length,
|
|
358
|
+
summary,
|
|
359
|
+
};
|
|
360
|
+
},
|
|
361
|
+
}),
|
|
362
|
+
};
|
|
363
|
+
}
|
|
364
|
+
findComponentTask(componentName) {
|
|
365
|
+
return this.currentPlan?.tests.find((t) => {
|
|
366
|
+
const ct = t;
|
|
367
|
+
return ct.component?.name === componentName || t.scenario.includes(componentName);
|
|
368
|
+
});
|
|
369
|
+
}
|
|
370
|
+
async saveToExperience(state, results) {
|
|
371
|
+
const experienceTracker = this.getExperienceTracker();
|
|
372
|
+
const actionResult = ActionResult.fromState(state);
|
|
373
|
+
const successfulInteractions = results.filter((r) => r.result === 'success' && r.code);
|
|
374
|
+
for (const interaction of successfulInteractions) {
|
|
375
|
+
await experienceTracker.saveSuccessfulResolution(actionResult, `Drill ${interaction.action}: ${interaction.component}`, interaction.code, interaction.description);
|
|
376
|
+
}
|
|
377
|
+
if (successfulInteractions.length > 0) {
|
|
378
|
+
tag('success').log(`Saved ${successfulInteractions.length} interactions to experience`);
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
async saveToKnowledge(knowledgePath, state, results) {
|
|
382
|
+
const knowledgeTracker = this.getKnowledgeTracker();
|
|
383
|
+
const successfulInteractions = results.filter((r) => r.result === 'success');
|
|
384
|
+
if (successfulInteractions.length === 0) {
|
|
385
|
+
tag('warning').log('No successful interactions to save to knowledge');
|
|
386
|
+
return;
|
|
387
|
+
}
|
|
388
|
+
const content = this.generateKnowledgeContent(state, successfulInteractions);
|
|
389
|
+
const result = knowledgeTracker.addKnowledge(knowledgePath, content);
|
|
390
|
+
tag('success').log(`Knowledge saved to: ${result.filePath}`);
|
|
391
|
+
}
|
|
392
|
+
generateKnowledgeContent(state, interactions) {
|
|
393
|
+
const lines = [];
|
|
394
|
+
lines.push('# Component Interactions\n');
|
|
395
|
+
lines.push(`Learned interactions from drilling ${state.url}\n`);
|
|
396
|
+
const groupedByComponent = new Map();
|
|
397
|
+
for (const interaction of interactions) {
|
|
398
|
+
const existing = groupedByComponent.get(interaction.component) || [];
|
|
399
|
+
existing.push(interaction);
|
|
400
|
+
groupedByComponent.set(interaction.component, existing);
|
|
401
|
+
}
|
|
402
|
+
for (const [component, items] of groupedByComponent) {
|
|
403
|
+
lines.push(`\n## ${component}\n`);
|
|
404
|
+
for (const item of items) {
|
|
405
|
+
lines.push(`- **${item.action}**: ${item.description}`);
|
|
406
|
+
if (item.code) {
|
|
407
|
+
lines.push('```js');
|
|
408
|
+
lines.push(item.code);
|
|
409
|
+
lines.push('```');
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
return lines.join('\n');
|
|
414
|
+
}
|
|
415
|
+
logSummary() {
|
|
416
|
+
if (!this.currentPlan)
|
|
417
|
+
return;
|
|
418
|
+
const total = this.currentPlan.tests.length;
|
|
419
|
+
const passed = this.currentPlan.tests.filter((t) => t.isSuccessful).length;
|
|
420
|
+
const failed = this.currentPlan.tests.filter((t) => t.hasFailed).length;
|
|
421
|
+
const successfulInteractions = this.allResults.filter((r) => r.result === 'success').length;
|
|
422
|
+
tag('info').log('\nDrill Summary:');
|
|
423
|
+
tag('info').log(` Total components: ${total}`);
|
|
424
|
+
tag('success').log(` Successful: ${passed}`);
|
|
425
|
+
if (failed > 0) {
|
|
426
|
+
tag('warning').log(` Failed: ${failed}`);
|
|
427
|
+
}
|
|
428
|
+
tag('info').log(` Total interactions learned: ${successfulInteractions}`);
|
|
429
|
+
for (const test of this.currentPlan.tests) {
|
|
430
|
+
const componentTask = test;
|
|
431
|
+
const status = test.isSuccessful ? '✓' : '✗';
|
|
432
|
+
const successCount = componentTask.interactions?.filter((i) => i.result === 'success').length || 0;
|
|
433
|
+
tag('step').log(` ${status} ${componentTask.component?.name || test.scenario}: ${successCount} interactions`);
|
|
434
|
+
}
|
|
435
|
+
}
|
|
436
|
+
getCurrentPlan() {
|
|
437
|
+
return this.currentPlan;
|
|
438
|
+
}
|
|
439
|
+
getConversation() {
|
|
440
|
+
return this.currentConversation;
|
|
441
|
+
}
|
|
442
|
+
}
|
|
443
|
+
//# sourceMappingURL=bosun.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bosun.js","sourceRoot":"","sources":["../../../src/ai/bosun.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { tool } from 'ai';
|
|
2
|
+
import { createBashTool } from 'bash-tool';
|
|
3
|
+
import dedent from 'dedent';
|
|
4
|
+
import { z } from 'zod';
|
|
5
|
+
import { ConfigParser } from "../../config.js";
|
|
6
|
+
import { Test } from "../../test-plan.js";
|
|
7
|
+
import { resolveProjectRoot } from "./mixin.js";
|
|
8
|
+
let cachedBashTool = null;
|
|
9
|
+
export function WithIdleMode(Base) {
|
|
10
|
+
return class extends Base {
|
|
11
|
+
async idleModeTools(ctx) {
|
|
12
|
+
const projectRoot = resolveProjectRoot();
|
|
13
|
+
const config = ConfigParser.getInstance().getConfig();
|
|
14
|
+
const knowledgeDir = config.dirs?.knowledge || 'knowledge';
|
|
15
|
+
const experienceDir = config.dirs?.experience || 'experience';
|
|
16
|
+
if (!cachedBashTool && projectRoot) {
|
|
17
|
+
cachedBashTool = await createBashTool({
|
|
18
|
+
destination: projectRoot,
|
|
19
|
+
onBeforeBashCall: ({ command }) => {
|
|
20
|
+
if (/\b(sudo|chmod|chown)\b/.test(command)) {
|
|
21
|
+
return { command: 'echo "Command not allowed" >&2 && exit 1' };
|
|
22
|
+
}
|
|
23
|
+
const writePatterns = [/>[^>]/, />>/, /\btee\b/, /\bmv\b/, /\bcp\b/, /\brm\b/];
|
|
24
|
+
if (writePatterns.some((p) => p.test(command)) && /\boutput[/\\]/.test(command)) {
|
|
25
|
+
return { command: 'echo "Write access to output/ is restricted" >&2 && exit 1' };
|
|
26
|
+
}
|
|
27
|
+
return { command };
|
|
28
|
+
},
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
const tools = {
|
|
32
|
+
updatePlan: tool({
|
|
33
|
+
description: 'Update the current plan by replacing or appending tests',
|
|
34
|
+
inputSchema: z.object({
|
|
35
|
+
action: z.enum(['replace', 'append']).optional().describe('replace clears existing tests, append keeps them'),
|
|
36
|
+
title: z.string().optional().describe('New plan title'),
|
|
37
|
+
tests: z
|
|
38
|
+
.array(z.object({
|
|
39
|
+
scenario: z.string(),
|
|
40
|
+
priority: z.enum(['critical', 'important', 'high', 'normal', 'low']).optional(),
|
|
41
|
+
expected: z.array(z.string()).optional(),
|
|
42
|
+
}))
|
|
43
|
+
.optional(),
|
|
44
|
+
}),
|
|
45
|
+
execute: async ({ action, title, tests }) => {
|
|
46
|
+
let plan = ctx.explorBot.getCurrentPlan();
|
|
47
|
+
if (!plan) {
|
|
48
|
+
plan = await ctx.explorBot.plan();
|
|
49
|
+
}
|
|
50
|
+
if (!plan) {
|
|
51
|
+
return { success: false, message: 'Plan unavailable' };
|
|
52
|
+
}
|
|
53
|
+
if (title) {
|
|
54
|
+
plan.title = title;
|
|
55
|
+
}
|
|
56
|
+
if (tests?.length) {
|
|
57
|
+
if (!action || action === 'replace') {
|
|
58
|
+
plan.tests.length = 0;
|
|
59
|
+
}
|
|
60
|
+
const currentUrl = ctx.explorBot.getExplorer().getStateManager().getCurrentState()?.url || '';
|
|
61
|
+
for (const testInput of tests) {
|
|
62
|
+
const priority = testInput.priority || 'normal';
|
|
63
|
+
const expected = testInput.expected?.length ? testInput.expected : [];
|
|
64
|
+
const test = new Test(testInput.scenario, priority, expected, currentUrl);
|
|
65
|
+
plan.addTest(test);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
plan.updateStatus();
|
|
69
|
+
return { success: true, tests: plan.tests.length };
|
|
70
|
+
},
|
|
71
|
+
}),
|
|
72
|
+
};
|
|
73
|
+
if (cachedBashTool) {
|
|
74
|
+
tools.bash = cachedBashTool.bash;
|
|
75
|
+
}
|
|
76
|
+
return tools;
|
|
77
|
+
}
|
|
78
|
+
idleModePrompt() {
|
|
79
|
+
const config = ConfigParser.getInstance().getConfig();
|
|
80
|
+
const knowledgeDir = config.dirs?.knowledge || 'knowledge';
|
|
81
|
+
const experienceDir = config.dirs?.experience || 'experience';
|
|
82
|
+
return dedent `
|
|
83
|
+
<idle_capabilities>
|
|
84
|
+
- Plan management: updatePlan() — replace or append tests in the current plan
|
|
85
|
+
- bash() — run shell commands for file operations
|
|
86
|
+
- READ from: ${knowledgeDir}/, ${experienceDir}/, output/
|
|
87
|
+
- WRITE to: ${knowledgeDir}/, ${experienceDir}/ only (NOT output/)
|
|
88
|
+
- Use ls to list files, cat to read small files
|
|
89
|
+
- Use head/tail for large files to avoid excessive output
|
|
90
|
+
- Use grep to search file contents
|
|
91
|
+
</idle_capabilities>
|
|
92
|
+
|
|
93
|
+
<knowledge_saving>
|
|
94
|
+
When user shares credentials, selectors, or important domain info during conversation,
|
|
95
|
+
suggest saving it to a knowledge file using bash tool.
|
|
96
|
+
Format: YAML frontmatter with url pattern, then content.
|
|
97
|
+
</knowledge_saving>
|
|
98
|
+
`;
|
|
99
|
+
}
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
//# sourceMappingURL=idle-mode.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"idle-mode.js","sourceRoot":"","sources":["../../../../src/ai/captain/idle-mode.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { dirname } from 'node:path';
|
|
2
|
+
import { ConfigParser } from "../../config.js";
|
|
3
|
+
import { createDebug } from '../../utils/logger.js';
|
|
4
|
+
export const debugLog = createDebug('explorbot:captain');
|
|
5
|
+
export function resolveProjectRoot() {
|
|
6
|
+
const configPath = ConfigParser.getInstance().getConfigPath();
|
|
7
|
+
if (!configPath)
|
|
8
|
+
return null;
|
|
9
|
+
return dirname(configPath);
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=mixin.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mixin.js","sourceRoot":"","sources":["../../../../src/ai/captain/mixin.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAG/C,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAIpD,MAAM,CAAC,MAAM,QAAQ,GAAG,WAAW,CAAC,mBAAmB,CAAC,CAAC;AASzD,MAAM,UAAU,kBAAkB;IAChC,MAAM,UAAU,GAAG,YAAY,CAAC,WAAW,EAAE,CAAC,aAAa,EAAE,CAAC;IAC9D,IAAI,CAAC,UAAU;QAAE,OAAO,IAAI,CAAC;IAC7B,OAAO,OAAO,CAAC,UAAU,CAAC,CAAC;AAC7B,CAAC"}
|