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,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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"}
|