@sentry/warden 0.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.agents/skills/find-bugs/SKILL.md +75 -0
- package/.agents/skills/vercel-react-best-practices/AGENTS.md +2934 -0
- package/.agents/skills/vercel-react-best-practices/SKILL.md +136 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-api-routes.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-dependencies.md +51 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-parallel.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-preload.md +50 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
- package/.claude/settings.json +57 -0
- package/.claude/settings.local.json +88 -0
- package/.claude/skills/agent-prompt/SKILL.md +54 -0
- package/.claude/skills/agent-prompt/references/agentic-patterns.md +94 -0
- package/.claude/skills/agent-prompt/references/anti-patterns.md +140 -0
- package/.claude/skills/agent-prompt/references/context-design.md +124 -0
- package/.claude/skills/agent-prompt/references/core-principles.md +75 -0
- package/.claude/skills/agent-prompt/references/model-guidance.md +118 -0
- package/.claude/skills/agent-prompt/references/output-formats.md +98 -0
- package/.claude/skills/agent-prompt/references/skill-structure.md +115 -0
- package/.claude/skills/agent-prompt/references/system-prompts.md +115 -0
- package/.claude/skills/notseer/SKILL.md +131 -0
- package/.claude/skills/skill-writer/SKILL.md +140 -0
- package/.claude/skills/testing-guidelines/SKILL.md +132 -0
- package/.claude/skills/warden-skill/SKILL.md +250 -0
- package/.claude/skills/warden-skill/references/config-schema.md +133 -0
- package/.dex/config.toml +2 -0
- package/.github/workflows/ci.yml +33 -0
- package/.github/workflows/release.yml +54 -0
- package/.github/workflows/warden.yml +40 -0
- package/AGENTS.md +89 -0
- package/CONTRIBUTING.md +60 -0
- package/LICENSE +105 -0
- package/README.md +43 -0
- package/SPEC.md +263 -0
- package/action.yml +87 -0
- package/assets/favicon.png +0 -0
- package/assets/warden-icon-bw.svg +5 -0
- package/assets/warden-icon-purple.png +0 -0
- package/assets/warden-icon-purple.svg +5 -0
- package/docs/.claude/settings.local.json +11 -0
- package/docs/astro.config.mjs +43 -0
- package/docs/package.json +19 -0
- package/docs/pnpm-lock.yaml +4000 -0
- package/docs/public/favicon.svg +5 -0
- package/docs/src/components/Code.astro +141 -0
- package/docs/src/components/PackageManagerTabs.astro +183 -0
- package/docs/src/components/Terminal.astro +212 -0
- package/docs/src/layouts/Base.astro +380 -0
- package/docs/src/pages/cli.astro +167 -0
- package/docs/src/pages/config.astro +394 -0
- package/docs/src/pages/guide.astro +449 -0
- package/docs/src/pages/index.astro +490 -0
- package/docs/src/styles/global.css +551 -0
- package/docs/tsconfig.json +3 -0
- package/docs/vercel.json +5 -0
- package/eslint.config.js +33 -0
- package/package.json +73 -0
- package/src/action/index.ts +1 -0
- package/src/action/main.ts +868 -0
- package/src/cli/args.test.ts +477 -0
- package/src/cli/args.ts +415 -0
- package/src/cli/commands/add.ts +447 -0
- package/src/cli/commands/init.test.ts +136 -0
- package/src/cli/commands/init.ts +132 -0
- package/src/cli/commands/setup-app/browser.ts +38 -0
- package/src/cli/commands/setup-app/credentials.ts +45 -0
- package/src/cli/commands/setup-app/manifest.ts +48 -0
- package/src/cli/commands/setup-app/server.ts +172 -0
- package/src/cli/commands/setup-app.ts +156 -0
- package/src/cli/commands/sync.ts +114 -0
- package/src/cli/context.ts +131 -0
- package/src/cli/files.test.ts +155 -0
- package/src/cli/files.ts +89 -0
- package/src/cli/fix.test.ts +310 -0
- package/src/cli/fix.ts +387 -0
- package/src/cli/git.test.ts +119 -0
- package/src/cli/git.ts +318 -0
- package/src/cli/index.ts +14 -0
- package/src/cli/main.ts +672 -0
- package/src/cli/output/box.ts +235 -0
- package/src/cli/output/formatters.test.ts +187 -0
- package/src/cli/output/formatters.ts +269 -0
- package/src/cli/output/icons.ts +13 -0
- package/src/cli/output/index.ts +44 -0
- package/src/cli/output/ink-runner.tsx +337 -0
- package/src/cli/output/jsonl.test.ts +347 -0
- package/src/cli/output/jsonl.ts +126 -0
- package/src/cli/output/reporter.ts +435 -0
- package/src/cli/output/tasks.ts +374 -0
- package/src/cli/output/tty.test.ts +117 -0
- package/src/cli/output/tty.ts +60 -0
- package/src/cli/output/verbosity.test.ts +40 -0
- package/src/cli/output/verbosity.ts +31 -0
- package/src/cli/terminal.test.ts +148 -0
- package/src/cli/terminal.ts +301 -0
- package/src/config/index.ts +3 -0
- package/src/config/loader.test.ts +313 -0
- package/src/config/loader.ts +103 -0
- package/src/config/schema.ts +168 -0
- package/src/config/writer.test.ts +119 -0
- package/src/config/writer.ts +84 -0
- package/src/diff/classify.test.ts +162 -0
- package/src/diff/classify.ts +92 -0
- package/src/diff/coalesce.test.ts +208 -0
- package/src/diff/coalesce.ts +133 -0
- package/src/diff/context.test.ts +226 -0
- package/src/diff/context.ts +201 -0
- package/src/diff/index.ts +4 -0
- package/src/diff/parser.test.ts +212 -0
- package/src/diff/parser.ts +149 -0
- package/src/event/context.ts +132 -0
- package/src/event/index.ts +2 -0
- package/src/event/schedule-context.ts +101 -0
- package/src/examples/examples.integration.test.ts +66 -0
- package/src/examples/index.test.ts +101 -0
- package/src/examples/index.ts +122 -0
- package/src/examples/setup.ts +25 -0
- package/src/index.ts +115 -0
- package/src/output/dedup.test.ts +419 -0
- package/src/output/dedup.ts +607 -0
- package/src/output/github-checks.test.ts +300 -0
- package/src/output/github-checks.ts +476 -0
- package/src/output/github-issues.ts +329 -0
- package/src/output/index.ts +5 -0
- package/src/output/issue-renderer.ts +197 -0
- package/src/output/renderer.test.ts +727 -0
- package/src/output/renderer.ts +217 -0
- package/src/output/stale.test.ts +375 -0
- package/src/output/stale.ts +155 -0
- package/src/output/types.ts +34 -0
- package/src/sdk/index.ts +1 -0
- package/src/sdk/runner.test.ts +806 -0
- package/src/sdk/runner.ts +1232 -0
- package/src/skills/index.ts +36 -0
- package/src/skills/loader.test.ts +300 -0
- package/src/skills/loader.ts +423 -0
- package/src/skills/remote.test.ts +704 -0
- package/src/skills/remote.ts +604 -0
- package/src/triggers/matcher.test.ts +277 -0
- package/src/triggers/matcher.ts +152 -0
- package/src/types/index.ts +194 -0
- package/src/utils/async.ts +18 -0
- package/src/utils/index.test.ts +84 -0
- package/src/utils/index.ts +50 -0
- package/tsconfig.json +25 -0
- package/vitest.config.ts +8 -0
- package/vitest.integration.config.ts +11 -0
- package/warden.toml +19 -0
|
@@ -0,0 +1,374 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Task execution for skills.
|
|
3
|
+
* Callback-based state updates for CLI and Ink rendering.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import type { SkillReport, SeverityThreshold, Finding, UsageStats, EventContext } from '../../types/index.js';
|
|
7
|
+
import type { SkillDefinition } from '../../config/schema.js';
|
|
8
|
+
import {
|
|
9
|
+
prepareFiles,
|
|
10
|
+
analyzeFile,
|
|
11
|
+
aggregateUsage,
|
|
12
|
+
deduplicateFindings,
|
|
13
|
+
generateSummary,
|
|
14
|
+
type SkillRunnerOptions,
|
|
15
|
+
type FileAnalysisCallbacks,
|
|
16
|
+
type PreparedFile,
|
|
17
|
+
type PRPromptContext,
|
|
18
|
+
} from '../../sdk/runner.js';
|
|
19
|
+
import chalk from 'chalk';
|
|
20
|
+
import figures from 'figures';
|
|
21
|
+
import { Verbosity } from './verbosity.js';
|
|
22
|
+
import type { OutputMode } from './tty.js';
|
|
23
|
+
import { ICON_CHECK, ICON_SKIPPED } from './icons.js';
|
|
24
|
+
import { formatDuration } from './formatters.js';
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* State of a file being processed by a skill.
|
|
28
|
+
*/
|
|
29
|
+
export interface FileState {
|
|
30
|
+
filename: string;
|
|
31
|
+
status: 'pending' | 'running' | 'done';
|
|
32
|
+
currentHunk: number;
|
|
33
|
+
totalHunks: number;
|
|
34
|
+
findings: Finding[];
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* State of a skill being executed.
|
|
39
|
+
*/
|
|
40
|
+
export interface SkillState {
|
|
41
|
+
name: string;
|
|
42
|
+
displayName: string;
|
|
43
|
+
status: 'pending' | 'running' | 'done' | 'skipped' | 'error';
|
|
44
|
+
startTime?: number;
|
|
45
|
+
durationMs?: number;
|
|
46
|
+
files: FileState[];
|
|
47
|
+
findings: Finding[];
|
|
48
|
+
usage?: UsageStats;
|
|
49
|
+
error?: string;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Result from running a skill task.
|
|
54
|
+
*/
|
|
55
|
+
export interface SkillTaskResult {
|
|
56
|
+
name: string;
|
|
57
|
+
report?: SkillReport;
|
|
58
|
+
failOn?: SeverityThreshold;
|
|
59
|
+
error?: unknown;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* Options for creating a skill task.
|
|
64
|
+
*/
|
|
65
|
+
export interface SkillTaskOptions {
|
|
66
|
+
name: string;
|
|
67
|
+
displayName?: string;
|
|
68
|
+
failOn?: SeverityThreshold;
|
|
69
|
+
/** Resolve the skill definition (may be async for loading) */
|
|
70
|
+
resolveSkill: () => Promise<SkillDefinition>;
|
|
71
|
+
/** The event context with files to analyze */
|
|
72
|
+
context: EventContext;
|
|
73
|
+
/** Options passed to the runner */
|
|
74
|
+
runnerOptions?: SkillRunnerOptions;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Options for running skill tasks.
|
|
79
|
+
*/
|
|
80
|
+
export interface RunTasksOptions {
|
|
81
|
+
mode: OutputMode;
|
|
82
|
+
verbosity: Verbosity;
|
|
83
|
+
concurrency: number;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Callbacks for reporting skill execution progress to the UI.
|
|
88
|
+
*/
|
|
89
|
+
export interface SkillProgressCallbacks {
|
|
90
|
+
onSkillStart: (skill: SkillState) => void;
|
|
91
|
+
onSkillUpdate: (name: string, updates: Partial<SkillState>) => void;
|
|
92
|
+
onFileUpdate: (skillName: string, filename: string, updates: Partial<FileState>) => void;
|
|
93
|
+
onSkillComplete: (name: string, report: SkillReport) => void;
|
|
94
|
+
onSkillSkipped: (name: string) => void;
|
|
95
|
+
onSkillError: (name: string, error: string) => void;
|
|
96
|
+
/** Called when a prompt exceeds the large prompt threshold */
|
|
97
|
+
onLargePrompt?: (skillName: string, filename: string, lineRange: string, chars: number, estimatedTokens: number) => void;
|
|
98
|
+
/** Called with prompt size info in debug mode */
|
|
99
|
+
onPromptSize?: (skillName: string, filename: string, lineRange: string, systemChars: number, userChars: number, totalChars: number, estimatedTokens: number) => void;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/**
|
|
103
|
+
* Run a single skill task.
|
|
104
|
+
*/
|
|
105
|
+
export async function runSkillTask(
|
|
106
|
+
options: SkillTaskOptions,
|
|
107
|
+
fileConcurrency: number,
|
|
108
|
+
callbacks: SkillProgressCallbacks
|
|
109
|
+
): Promise<SkillTaskResult> {
|
|
110
|
+
const { name, displayName = name, failOn, resolveSkill, context, runnerOptions = {} } = options;
|
|
111
|
+
const startTime = Date.now();
|
|
112
|
+
|
|
113
|
+
try {
|
|
114
|
+
// Resolve the skill
|
|
115
|
+
const skill = await resolveSkill();
|
|
116
|
+
|
|
117
|
+
// Prepare files (parse patches into hunks)
|
|
118
|
+
const { files: preparedFiles, skippedFiles } = prepareFiles(context, {
|
|
119
|
+
contextLines: runnerOptions.contextLines,
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
if (preparedFiles.length === 0) {
|
|
123
|
+
// No files to analyze - skip
|
|
124
|
+
callbacks.onSkillSkipped(name);
|
|
125
|
+
return {
|
|
126
|
+
name,
|
|
127
|
+
report: {
|
|
128
|
+
skill: skill.name,
|
|
129
|
+
summary: 'No code changes to analyze',
|
|
130
|
+
findings: [],
|
|
131
|
+
usage: { inputTokens: 0, outputTokens: 0, costUSD: 0 },
|
|
132
|
+
skippedFiles: skippedFiles.length > 0 ? skippedFiles : undefined,
|
|
133
|
+
},
|
|
134
|
+
failOn,
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
// Initialize file states
|
|
139
|
+
const fileStates: FileState[] = preparedFiles.map((file) => ({
|
|
140
|
+
filename: file.filename,
|
|
141
|
+
status: 'pending',
|
|
142
|
+
currentHunk: 0,
|
|
143
|
+
totalHunks: file.hunks.length,
|
|
144
|
+
findings: [],
|
|
145
|
+
}));
|
|
146
|
+
|
|
147
|
+
// Notify skill start
|
|
148
|
+
callbacks.onSkillStart({
|
|
149
|
+
name,
|
|
150
|
+
displayName,
|
|
151
|
+
status: 'running',
|
|
152
|
+
startTime,
|
|
153
|
+
files: fileStates,
|
|
154
|
+
findings: [],
|
|
155
|
+
});
|
|
156
|
+
|
|
157
|
+
// Build PR context for inclusion in prompts (if available)
|
|
158
|
+
const prContext: PRPromptContext | undefined = context.pullRequest
|
|
159
|
+
? {
|
|
160
|
+
changedFiles: context.pullRequest.files.map((f) => f.filename),
|
|
161
|
+
title: context.pullRequest.title,
|
|
162
|
+
body: context.pullRequest.body,
|
|
163
|
+
}
|
|
164
|
+
: undefined;
|
|
165
|
+
|
|
166
|
+
// Process files with concurrency
|
|
167
|
+
const processFile = async (prepared: PreparedFile, index: number): Promise<{ findings: Finding[]; usage?: UsageStats; failedHunks: number }> => {
|
|
168
|
+
const filename = prepared.filename;
|
|
169
|
+
|
|
170
|
+
// Update file state to running
|
|
171
|
+
callbacks.onFileUpdate(name, filename, { status: 'running' });
|
|
172
|
+
|
|
173
|
+
const fileCallbacks: FileAnalysisCallbacks = {
|
|
174
|
+
skillStartTime: startTime,
|
|
175
|
+
onHunkStart: (hunkNum, totalHunks) => {
|
|
176
|
+
callbacks.onFileUpdate(name, filename, {
|
|
177
|
+
currentHunk: hunkNum,
|
|
178
|
+
totalHunks,
|
|
179
|
+
});
|
|
180
|
+
},
|
|
181
|
+
onHunkComplete: (_hunkNum, findings) => {
|
|
182
|
+
// Accumulate findings for this file
|
|
183
|
+
const current = fileStates[index];
|
|
184
|
+
if (current) {
|
|
185
|
+
current.findings.push(...findings);
|
|
186
|
+
}
|
|
187
|
+
},
|
|
188
|
+
onLargePrompt: callbacks.onLargePrompt
|
|
189
|
+
? (lineRange, chars, estimatedTokens) => {
|
|
190
|
+
callbacks.onLargePrompt?.(name, filename, lineRange, chars, estimatedTokens);
|
|
191
|
+
}
|
|
192
|
+
: undefined,
|
|
193
|
+
onPromptSize: callbacks.onPromptSize
|
|
194
|
+
? (lineRange, systemChars, userChars, totalChars, estimatedTokens) => {
|
|
195
|
+
callbacks.onPromptSize?.(name, filename, lineRange, systemChars, userChars, totalChars, estimatedTokens);
|
|
196
|
+
}
|
|
197
|
+
: undefined,
|
|
198
|
+
};
|
|
199
|
+
|
|
200
|
+
const result = await analyzeFile(
|
|
201
|
+
skill,
|
|
202
|
+
prepared,
|
|
203
|
+
context.repoPath,
|
|
204
|
+
runnerOptions,
|
|
205
|
+
fileCallbacks,
|
|
206
|
+
prContext
|
|
207
|
+
);
|
|
208
|
+
|
|
209
|
+
// Update file state to done
|
|
210
|
+
callbacks.onFileUpdate(name, filename, {
|
|
211
|
+
status: 'done',
|
|
212
|
+
findings: result.findings,
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
return { findings: result.findings, usage: result.usage, failedHunks: result.failedHunks };
|
|
216
|
+
};
|
|
217
|
+
|
|
218
|
+
// Process files in batches with concurrency
|
|
219
|
+
const allResults: { findings: Finding[]; usage?: UsageStats; failedHunks: number }[] = [];
|
|
220
|
+
|
|
221
|
+
for (let i = 0; i < preparedFiles.length; i += fileConcurrency) {
|
|
222
|
+
const batch = preparedFiles.slice(i, i + fileConcurrency);
|
|
223
|
+
const batchResults = await Promise.all(
|
|
224
|
+
batch.map((file, batchIndex) => processFile(file, i + batchIndex))
|
|
225
|
+
);
|
|
226
|
+
allResults.push(...batchResults);
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
// Build report
|
|
230
|
+
const duration = Date.now() - startTime;
|
|
231
|
+
const allFindings = allResults.flatMap((r) => r.findings);
|
|
232
|
+
const allUsage = allResults.map((r) => r.usage).filter((u): u is UsageStats => u !== undefined);
|
|
233
|
+
const totalFailedHunks = allResults.reduce((sum, r) => sum + r.failedHunks, 0);
|
|
234
|
+
const uniqueFindings = deduplicateFindings(allFindings);
|
|
235
|
+
|
|
236
|
+
const report: SkillReport = {
|
|
237
|
+
skill: skill.name,
|
|
238
|
+
summary: generateSummary(skill.name, uniqueFindings),
|
|
239
|
+
findings: uniqueFindings,
|
|
240
|
+
usage: aggregateUsage(allUsage),
|
|
241
|
+
durationMs: duration,
|
|
242
|
+
};
|
|
243
|
+
if (skippedFiles.length > 0) {
|
|
244
|
+
report.skippedFiles = skippedFiles;
|
|
245
|
+
}
|
|
246
|
+
if (totalFailedHunks > 0) {
|
|
247
|
+
report.failedHunks = totalFailedHunks;
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
// Notify skill complete
|
|
251
|
+
callbacks.onSkillUpdate(name, {
|
|
252
|
+
status: 'done',
|
|
253
|
+
durationMs: duration,
|
|
254
|
+
findings: uniqueFindings,
|
|
255
|
+
usage: report.usage,
|
|
256
|
+
});
|
|
257
|
+
callbacks.onSkillComplete(name, report);
|
|
258
|
+
|
|
259
|
+
return { name, report, failOn };
|
|
260
|
+
} catch (err) {
|
|
261
|
+
const errorMessage = err instanceof Error ? err.message : String(err);
|
|
262
|
+
callbacks.onSkillError(name, errorMessage);
|
|
263
|
+
return { name, error: err, failOn };
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
/**
|
|
268
|
+
* Run multiple skill tasks with optional concurrency.
|
|
269
|
+
* Uses callbacks to report progress for Ink rendering.
|
|
270
|
+
*/
|
|
271
|
+
export async function runSkillTasks(
|
|
272
|
+
tasks: SkillTaskOptions[],
|
|
273
|
+
options: RunTasksOptions,
|
|
274
|
+
callbacks?: SkillProgressCallbacks
|
|
275
|
+
): Promise<SkillTaskResult[]> {
|
|
276
|
+
const { mode, verbosity, concurrency } = options;
|
|
277
|
+
|
|
278
|
+
// File-level concurrency (within each skill)
|
|
279
|
+
const fileConcurrency = 5;
|
|
280
|
+
|
|
281
|
+
// Create default callbacks that output to console
|
|
282
|
+
const defaultCallbacks: SkillProgressCallbacks = {
|
|
283
|
+
onSkillStart: (_skill) => {
|
|
284
|
+
// We don't log start - we'll log completion with duration
|
|
285
|
+
},
|
|
286
|
+
onSkillUpdate: () => { /* no-op for default callbacks */ },
|
|
287
|
+
onFileUpdate: () => { /* no-op for default callbacks */ },
|
|
288
|
+
onSkillComplete: (name, report) => {
|
|
289
|
+
if (verbosity === Verbosity.Quiet) return;
|
|
290
|
+
const task = tasks.find((t) => t.name === name);
|
|
291
|
+
const displayName = task?.displayName ?? name;
|
|
292
|
+
const duration = report.durationMs ? ` ${chalk.dim(`[${formatDuration(report.durationMs)}]`)}` : '';
|
|
293
|
+
if (mode.isTTY) {
|
|
294
|
+
console.error(`${chalk.green(ICON_CHECK)} ${displayName}${duration}`);
|
|
295
|
+
} else {
|
|
296
|
+
console.log(`${ICON_CHECK} ${displayName}`);
|
|
297
|
+
}
|
|
298
|
+
},
|
|
299
|
+
onSkillSkipped: (name) => {
|
|
300
|
+
if (verbosity === Verbosity.Quiet) return;
|
|
301
|
+
const task = tasks.find((t) => t.name === name);
|
|
302
|
+
const displayName = task?.displayName ?? name;
|
|
303
|
+
if (mode.isTTY) {
|
|
304
|
+
console.error(`${chalk.yellow(ICON_SKIPPED)} ${displayName} ${chalk.dim('[skipped]')}`);
|
|
305
|
+
} else {
|
|
306
|
+
console.log(`${ICON_SKIPPED} ${displayName} [skipped]`);
|
|
307
|
+
}
|
|
308
|
+
},
|
|
309
|
+
onSkillError: (name, error) => {
|
|
310
|
+
if (verbosity === Verbosity.Quiet) return;
|
|
311
|
+
const task = tasks.find((t) => t.name === name);
|
|
312
|
+
const displayName = task?.displayName ?? name;
|
|
313
|
+
if (mode.isTTY) {
|
|
314
|
+
console.error(`${chalk.red('\u2717')} ${displayName} - ${chalk.red(error)}`);
|
|
315
|
+
} else {
|
|
316
|
+
console.error(`\u2717 ${displayName} - Error: ${error}`);
|
|
317
|
+
}
|
|
318
|
+
},
|
|
319
|
+
// Warn about large prompts (always shown unless quiet)
|
|
320
|
+
onLargePrompt: (skillName, filename, lineRange, chars, estimatedTokens) => {
|
|
321
|
+
if (verbosity === Verbosity.Quiet) return;
|
|
322
|
+
const location = `${filename}:${lineRange}`;
|
|
323
|
+
const size = `${Math.round(chars / 1000)}k chars (~${Math.round(estimatedTokens / 1000)}k tokens)`;
|
|
324
|
+
if (mode.isTTY) {
|
|
325
|
+
console.error(`${chalk.yellow(figures.warning)} Large prompt for ${location}: ${size}`);
|
|
326
|
+
} else {
|
|
327
|
+
console.error(`WARN: Large prompt for ${location}: ${size}`);
|
|
328
|
+
}
|
|
329
|
+
},
|
|
330
|
+
// Debug mode: show prompt sizes
|
|
331
|
+
onPromptSize: verbosity >= Verbosity.Debug
|
|
332
|
+
? (_skillName, filename, lineRange, systemChars, userChars, totalChars, estimatedTokens) => {
|
|
333
|
+
const location = `${filename}:${lineRange}`;
|
|
334
|
+
if (mode.isTTY) {
|
|
335
|
+
console.error(chalk.dim(`[debug] Prompt for ${location}: system=${systemChars}, user=${userChars}, total=${totalChars} chars (~${estimatedTokens} tokens)`));
|
|
336
|
+
} else {
|
|
337
|
+
console.error(`DEBUG: Prompt for ${location}: system=${systemChars}, user=${userChars}, total=${totalChars} chars (~${estimatedTokens} tokens)`);
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
: undefined,
|
|
341
|
+
};
|
|
342
|
+
|
|
343
|
+
const effectiveCallbacks = callbacks ?? defaultCallbacks;
|
|
344
|
+
|
|
345
|
+
// Output SKILLS header
|
|
346
|
+
if (verbosity !== Verbosity.Quiet && tasks.length > 0) {
|
|
347
|
+
if (mode.isTTY) {
|
|
348
|
+
console.error(chalk.bold('SKILLS'));
|
|
349
|
+
} else {
|
|
350
|
+
console.error('SKILLS');
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
const results: SkillTaskResult[] = [];
|
|
355
|
+
|
|
356
|
+
if (concurrency <= 1) {
|
|
357
|
+
// Sequential execution
|
|
358
|
+
for (const task of tasks) {
|
|
359
|
+
const result = await runSkillTask(task, fileConcurrency, effectiveCallbacks);
|
|
360
|
+
results.push(result);
|
|
361
|
+
}
|
|
362
|
+
} else {
|
|
363
|
+
// Parallel execution with concurrency limit
|
|
364
|
+
for (let i = 0; i < tasks.length; i += concurrency) {
|
|
365
|
+
const batch = tasks.slice(i, i + concurrency);
|
|
366
|
+
const batchResults = await Promise.all(
|
|
367
|
+
batch.map((task) => runSkillTask(task, fileConcurrency, effectiveCallbacks))
|
|
368
|
+
);
|
|
369
|
+
results.push(...batchResults);
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
return results;
|
|
374
|
+
}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { detectOutputMode } from './tty.js';
|
|
3
|
+
|
|
4
|
+
describe('detectOutputMode', () => {
|
|
5
|
+
const originalEnv = process.env;
|
|
6
|
+
const originalStderr = process.stderr.isTTY;
|
|
7
|
+
const originalStdout = process.stdout.isTTY;
|
|
8
|
+
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
process.env = { ...originalEnv };
|
|
11
|
+
// Reset to non-TTY by default for predictable tests
|
|
12
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
13
|
+
value: false,
|
|
14
|
+
configurable: true,
|
|
15
|
+
});
|
|
16
|
+
Object.defineProperty(process.stdout, 'isTTY', {
|
|
17
|
+
value: false,
|
|
18
|
+
configurable: true,
|
|
19
|
+
});
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
afterEach(() => {
|
|
23
|
+
process.env = originalEnv;
|
|
24
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
25
|
+
value: originalStderr,
|
|
26
|
+
configurable: true,
|
|
27
|
+
});
|
|
28
|
+
Object.defineProperty(process.stdout, 'isTTY', {
|
|
29
|
+
value: originalStdout,
|
|
30
|
+
configurable: true,
|
|
31
|
+
});
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('returns isTTY=false when streams are not TTY', () => {
|
|
35
|
+
process.env['TERM'] = 'xterm-256color';
|
|
36
|
+
const mode = detectOutputMode();
|
|
37
|
+
expect(mode.isTTY).toBe(false);
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it('returns isTTY=true when stderr is TTY with valid TERM', () => {
|
|
41
|
+
process.env['TERM'] = 'xterm-256color';
|
|
42
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
43
|
+
value: true,
|
|
44
|
+
configurable: true,
|
|
45
|
+
});
|
|
46
|
+
const mode = detectOutputMode();
|
|
47
|
+
expect(mode.isTTY).toBe(true);
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it('returns isTTY=false when TERM=dumb even if stream is TTY', () => {
|
|
51
|
+
process.env['TERM'] = 'dumb';
|
|
52
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
53
|
+
value: true,
|
|
54
|
+
configurable: true,
|
|
55
|
+
});
|
|
56
|
+
const mode = detectOutputMode();
|
|
57
|
+
expect(mode.isTTY).toBe(false);
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it('returns isTTY=false when TERM is empty even if stream is TTY', () => {
|
|
61
|
+
process.env['TERM'] = '';
|
|
62
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
63
|
+
value: true,
|
|
64
|
+
configurable: true,
|
|
65
|
+
});
|
|
66
|
+
const mode = detectOutputMode();
|
|
67
|
+
expect(mode.isTTY).toBe(false);
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
it('returns isTTY=false when TERM is unset even if stream is TTY', () => {
|
|
71
|
+
delete process.env['TERM'];
|
|
72
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
73
|
+
value: true,
|
|
74
|
+
configurable: true,
|
|
75
|
+
});
|
|
76
|
+
const mode = detectOutputMode();
|
|
77
|
+
expect(mode.isTTY).toBe(false);
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it('respects NO_COLOR environment variable', () => {
|
|
81
|
+
process.env['TERM'] = 'xterm-256color';
|
|
82
|
+
process.env['NO_COLOR'] = '1';
|
|
83
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
84
|
+
value: true,
|
|
85
|
+
configurable: true,
|
|
86
|
+
});
|
|
87
|
+
const mode = detectOutputMode();
|
|
88
|
+
expect(mode.supportsColor).toBe(false);
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
it('respects FORCE_COLOR environment variable', () => {
|
|
92
|
+
process.env['TERM'] = 'xterm-256color';
|
|
93
|
+
process.env['FORCE_COLOR'] = '1';
|
|
94
|
+
const mode = detectOutputMode();
|
|
95
|
+
expect(mode.supportsColor).toBe(true);
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
it('respects colorOverride=true', () => {
|
|
99
|
+
const mode = detectOutputMode(true);
|
|
100
|
+
expect(mode.supportsColor).toBe(true);
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
it('respects colorOverride=false', () => {
|
|
104
|
+
process.env['TERM'] = 'xterm-256color';
|
|
105
|
+
Object.defineProperty(process.stderr, 'isTTY', {
|
|
106
|
+
value: true,
|
|
107
|
+
configurable: true,
|
|
108
|
+
});
|
|
109
|
+
const mode = detectOutputMode(false);
|
|
110
|
+
expect(mode.supportsColor).toBe(false);
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it('defaults columns to 80', () => {
|
|
114
|
+
const mode = detectOutputMode();
|
|
115
|
+
expect(mode.columns).toBe(80);
|
|
116
|
+
});
|
|
117
|
+
});
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Output mode configuration based on terminal capabilities.
|
|
5
|
+
*/
|
|
6
|
+
export interface OutputMode {
|
|
7
|
+
/** Whether stdout is a TTY */
|
|
8
|
+
isTTY: boolean;
|
|
9
|
+
/** Whether colors are supported */
|
|
10
|
+
supportsColor: boolean;
|
|
11
|
+
/** Terminal width in columns */
|
|
12
|
+
columns: number;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Detect terminal capabilities.
|
|
17
|
+
* @param colorOverride - Optional override for color support (--color / --no-color)
|
|
18
|
+
*/
|
|
19
|
+
export function detectOutputMode(colorOverride?: boolean): OutputMode {
|
|
20
|
+
// Check both stderr and stdout for TTY - some environments have TTY on one but not the other
|
|
21
|
+
const streamIsTTY = (process.stderr.isTTY || process.stdout.isTTY) ?? false;
|
|
22
|
+
|
|
23
|
+
// Treat dumb terminals as non-TTY (e.g., TERM=dumb used by some editors/agents)
|
|
24
|
+
const term = process.env['TERM'] ?? '';
|
|
25
|
+
const isDumbTerminal = term === 'dumb' || term === '';
|
|
26
|
+
|
|
27
|
+
const isTTY = streamIsTTY && !isDumbTerminal;
|
|
28
|
+
|
|
29
|
+
// Determine color support
|
|
30
|
+
let supportsColor: boolean;
|
|
31
|
+
if (colorOverride !== undefined) {
|
|
32
|
+
supportsColor = colorOverride;
|
|
33
|
+
} else if (process.env['NO_COLOR']) {
|
|
34
|
+
supportsColor = false;
|
|
35
|
+
} else if (process.env['FORCE_COLOR']) {
|
|
36
|
+
supportsColor = true;
|
|
37
|
+
} else {
|
|
38
|
+
supportsColor = isTTY && chalk.level > 0;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// Configure chalk based on color support
|
|
42
|
+
if (!supportsColor) {
|
|
43
|
+
chalk.level = 0;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
const columns = process.stderr.columns ?? process.stdout.columns ?? 80;
|
|
47
|
+
|
|
48
|
+
return {
|
|
49
|
+
isTTY,
|
|
50
|
+
supportsColor,
|
|
51
|
+
columns,
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Get a timestamp for CI/non-TTY output.
|
|
57
|
+
*/
|
|
58
|
+
export function timestamp(): string {
|
|
59
|
+
return new Date().toISOString();
|
|
60
|
+
}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { Verbosity, parseVerbosity } from './verbosity.js';
|
|
3
|
+
|
|
4
|
+
describe('parseVerbosity', () => {
|
|
5
|
+
it('returns Quiet when quiet is true', () => {
|
|
6
|
+
expect(parseVerbosity(true, 0)).toBe(Verbosity.Quiet);
|
|
7
|
+
expect(parseVerbosity(true, 1)).toBe(Verbosity.Quiet);
|
|
8
|
+
expect(parseVerbosity(true, 2)).toBe(Verbosity.Quiet);
|
|
9
|
+
});
|
|
10
|
+
|
|
11
|
+
it('returns Normal when verbose count is 0', () => {
|
|
12
|
+
expect(parseVerbosity(false, 0)).toBe(Verbosity.Normal);
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
it('returns Verbose when verbose count is 1', () => {
|
|
16
|
+
expect(parseVerbosity(false, 1)).toBe(Verbosity.Verbose);
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
it('returns Debug when verbose count is 2 or more', () => {
|
|
20
|
+
expect(parseVerbosity(false, 2)).toBe(Verbosity.Debug);
|
|
21
|
+
expect(parseVerbosity(false, 3)).toBe(Verbosity.Debug);
|
|
22
|
+
expect(parseVerbosity(false, 10)).toBe(Verbosity.Debug);
|
|
23
|
+
});
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
describe('Verbosity enum', () => {
|
|
27
|
+
it('has correct numeric values', () => {
|
|
28
|
+
expect(Verbosity.Quiet).toBe(0);
|
|
29
|
+
expect(Verbosity.Normal).toBe(1);
|
|
30
|
+
expect(Verbosity.Verbose).toBe(2);
|
|
31
|
+
expect(Verbosity.Debug).toBe(3);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('supports comparison operators', () => {
|
|
35
|
+
expect(Verbosity.Quiet < Verbosity.Normal).toBe(true);
|
|
36
|
+
expect(Verbosity.Normal < Verbosity.Verbose).toBe(true);
|
|
37
|
+
expect(Verbosity.Verbose < Verbosity.Debug).toBe(true);
|
|
38
|
+
expect(Verbosity.Debug >= Verbosity.Verbose).toBe(true);
|
|
39
|
+
});
|
|
40
|
+
});
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Verbosity levels for CLI output.
|
|
3
|
+
*/
|
|
4
|
+
export enum Verbosity {
|
|
5
|
+
/** Errors + final summary only */
|
|
6
|
+
Quiet = 0,
|
|
7
|
+
/** Normal output with progress */
|
|
8
|
+
Normal = 1,
|
|
9
|
+
/** Real-time findings, hunk details */
|
|
10
|
+
Verbose = 2,
|
|
11
|
+
/** Token counts, latencies, debug info */
|
|
12
|
+
Debug = 3,
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Parse verbosity from CLI flags.
|
|
17
|
+
* @param quiet - If true, return Quiet
|
|
18
|
+
* @param verboseCount - Number of -v flags (0, 1, or 2+)
|
|
19
|
+
*/
|
|
20
|
+
export function parseVerbosity(quiet: boolean, verboseCount: number): Verbosity {
|
|
21
|
+
if (quiet) {
|
|
22
|
+
return Verbosity.Quiet;
|
|
23
|
+
}
|
|
24
|
+
if (verboseCount >= 2) {
|
|
25
|
+
return Verbosity.Debug;
|
|
26
|
+
}
|
|
27
|
+
if (verboseCount === 1) {
|
|
28
|
+
return Verbosity.Verbose;
|
|
29
|
+
}
|
|
30
|
+
return Verbosity.Normal;
|
|
31
|
+
}
|