@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.
Files changed (199) hide show
  1. package/.agents/skills/find-bugs/SKILL.md +75 -0
  2. package/.agents/skills/vercel-react-best-practices/AGENTS.md +2934 -0
  3. package/.agents/skills/vercel-react-best-practices/SKILL.md +136 -0
  4. package/.agents/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
  5. package/.agents/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
  6. package/.agents/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
  7. package/.agents/skills/vercel-react-best-practices/rules/async-api-routes.md +38 -0
  8. package/.agents/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
  9. package/.agents/skills/vercel-react-best-practices/rules/async-dependencies.md +51 -0
  10. package/.agents/skills/vercel-react-best-practices/rules/async-parallel.md +28 -0
  11. package/.agents/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
  12. package/.agents/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
  13. package/.agents/skills/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
  14. package/.agents/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
  15. package/.agents/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
  16. package/.agents/skills/vercel-react-best-practices/rules/bundle-preload.md +50 -0
  17. package/.agents/skills/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
  18. package/.agents/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
  19. package/.agents/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
  20. package/.agents/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
  21. package/.agents/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
  22. package/.agents/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
  23. package/.agents/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
  24. package/.agents/skills/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
  25. package/.agents/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
  26. package/.agents/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
  27. package/.agents/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
  28. package/.agents/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
  29. package/.agents/skills/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
  30. package/.agents/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
  31. package/.agents/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
  32. package/.agents/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
  33. package/.agents/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
  34. package/.agents/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
  35. package/.agents/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
  36. package/.agents/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
  37. package/.agents/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
  38. package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
  39. package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
  40. package/.agents/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
  41. package/.agents/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
  42. package/.agents/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
  43. package/.agents/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
  44. package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
  45. package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
  46. package/.agents/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
  47. package/.agents/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
  48. package/.agents/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
  49. package/.agents/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
  50. package/.agents/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
  51. package/.agents/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
  52. package/.agents/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
  53. package/.agents/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
  54. package/.agents/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
  55. package/.agents/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
  56. package/.agents/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
  57. package/.agents/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
  58. package/.agents/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
  59. package/.agents/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
  60. package/.agents/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
  61. package/.claude/settings.json +57 -0
  62. package/.claude/settings.local.json +88 -0
  63. package/.claude/skills/agent-prompt/SKILL.md +54 -0
  64. package/.claude/skills/agent-prompt/references/agentic-patterns.md +94 -0
  65. package/.claude/skills/agent-prompt/references/anti-patterns.md +140 -0
  66. package/.claude/skills/agent-prompt/references/context-design.md +124 -0
  67. package/.claude/skills/agent-prompt/references/core-principles.md +75 -0
  68. package/.claude/skills/agent-prompt/references/model-guidance.md +118 -0
  69. package/.claude/skills/agent-prompt/references/output-formats.md +98 -0
  70. package/.claude/skills/agent-prompt/references/skill-structure.md +115 -0
  71. package/.claude/skills/agent-prompt/references/system-prompts.md +115 -0
  72. package/.claude/skills/notseer/SKILL.md +131 -0
  73. package/.claude/skills/skill-writer/SKILL.md +140 -0
  74. package/.claude/skills/testing-guidelines/SKILL.md +132 -0
  75. package/.claude/skills/warden-skill/SKILL.md +250 -0
  76. package/.claude/skills/warden-skill/references/config-schema.md +133 -0
  77. package/.dex/config.toml +2 -0
  78. package/.github/workflows/ci.yml +33 -0
  79. package/.github/workflows/release.yml +54 -0
  80. package/.github/workflows/warden.yml +40 -0
  81. package/AGENTS.md +89 -0
  82. package/CONTRIBUTING.md +60 -0
  83. package/LICENSE +105 -0
  84. package/README.md +43 -0
  85. package/SPEC.md +263 -0
  86. package/action.yml +87 -0
  87. package/assets/favicon.png +0 -0
  88. package/assets/warden-icon-bw.svg +5 -0
  89. package/assets/warden-icon-purple.png +0 -0
  90. package/assets/warden-icon-purple.svg +5 -0
  91. package/docs/.claude/settings.local.json +11 -0
  92. package/docs/astro.config.mjs +43 -0
  93. package/docs/package.json +19 -0
  94. package/docs/pnpm-lock.yaml +4000 -0
  95. package/docs/public/favicon.svg +5 -0
  96. package/docs/src/components/Code.astro +141 -0
  97. package/docs/src/components/PackageManagerTabs.astro +183 -0
  98. package/docs/src/components/Terminal.astro +212 -0
  99. package/docs/src/layouts/Base.astro +380 -0
  100. package/docs/src/pages/cli.astro +167 -0
  101. package/docs/src/pages/config.astro +394 -0
  102. package/docs/src/pages/guide.astro +449 -0
  103. package/docs/src/pages/index.astro +490 -0
  104. package/docs/src/styles/global.css +551 -0
  105. package/docs/tsconfig.json +3 -0
  106. package/docs/vercel.json +5 -0
  107. package/eslint.config.js +33 -0
  108. package/package.json +73 -0
  109. package/src/action/index.ts +1 -0
  110. package/src/action/main.ts +868 -0
  111. package/src/cli/args.test.ts +477 -0
  112. package/src/cli/args.ts +415 -0
  113. package/src/cli/commands/add.ts +447 -0
  114. package/src/cli/commands/init.test.ts +136 -0
  115. package/src/cli/commands/init.ts +132 -0
  116. package/src/cli/commands/setup-app/browser.ts +38 -0
  117. package/src/cli/commands/setup-app/credentials.ts +45 -0
  118. package/src/cli/commands/setup-app/manifest.ts +48 -0
  119. package/src/cli/commands/setup-app/server.ts +172 -0
  120. package/src/cli/commands/setup-app.ts +156 -0
  121. package/src/cli/commands/sync.ts +114 -0
  122. package/src/cli/context.ts +131 -0
  123. package/src/cli/files.test.ts +155 -0
  124. package/src/cli/files.ts +89 -0
  125. package/src/cli/fix.test.ts +310 -0
  126. package/src/cli/fix.ts +387 -0
  127. package/src/cli/git.test.ts +119 -0
  128. package/src/cli/git.ts +318 -0
  129. package/src/cli/index.ts +14 -0
  130. package/src/cli/main.ts +672 -0
  131. package/src/cli/output/box.ts +235 -0
  132. package/src/cli/output/formatters.test.ts +187 -0
  133. package/src/cli/output/formatters.ts +269 -0
  134. package/src/cli/output/icons.ts +13 -0
  135. package/src/cli/output/index.ts +44 -0
  136. package/src/cli/output/ink-runner.tsx +337 -0
  137. package/src/cli/output/jsonl.test.ts +347 -0
  138. package/src/cli/output/jsonl.ts +126 -0
  139. package/src/cli/output/reporter.ts +435 -0
  140. package/src/cli/output/tasks.ts +374 -0
  141. package/src/cli/output/tty.test.ts +117 -0
  142. package/src/cli/output/tty.ts +60 -0
  143. package/src/cli/output/verbosity.test.ts +40 -0
  144. package/src/cli/output/verbosity.ts +31 -0
  145. package/src/cli/terminal.test.ts +148 -0
  146. package/src/cli/terminal.ts +301 -0
  147. package/src/config/index.ts +3 -0
  148. package/src/config/loader.test.ts +313 -0
  149. package/src/config/loader.ts +103 -0
  150. package/src/config/schema.ts +168 -0
  151. package/src/config/writer.test.ts +119 -0
  152. package/src/config/writer.ts +84 -0
  153. package/src/diff/classify.test.ts +162 -0
  154. package/src/diff/classify.ts +92 -0
  155. package/src/diff/coalesce.test.ts +208 -0
  156. package/src/diff/coalesce.ts +133 -0
  157. package/src/diff/context.test.ts +226 -0
  158. package/src/diff/context.ts +201 -0
  159. package/src/diff/index.ts +4 -0
  160. package/src/diff/parser.test.ts +212 -0
  161. package/src/diff/parser.ts +149 -0
  162. package/src/event/context.ts +132 -0
  163. package/src/event/index.ts +2 -0
  164. package/src/event/schedule-context.ts +101 -0
  165. package/src/examples/examples.integration.test.ts +66 -0
  166. package/src/examples/index.test.ts +101 -0
  167. package/src/examples/index.ts +122 -0
  168. package/src/examples/setup.ts +25 -0
  169. package/src/index.ts +115 -0
  170. package/src/output/dedup.test.ts +419 -0
  171. package/src/output/dedup.ts +607 -0
  172. package/src/output/github-checks.test.ts +300 -0
  173. package/src/output/github-checks.ts +476 -0
  174. package/src/output/github-issues.ts +329 -0
  175. package/src/output/index.ts +5 -0
  176. package/src/output/issue-renderer.ts +197 -0
  177. package/src/output/renderer.test.ts +727 -0
  178. package/src/output/renderer.ts +217 -0
  179. package/src/output/stale.test.ts +375 -0
  180. package/src/output/stale.ts +155 -0
  181. package/src/output/types.ts +34 -0
  182. package/src/sdk/index.ts +1 -0
  183. package/src/sdk/runner.test.ts +806 -0
  184. package/src/sdk/runner.ts +1232 -0
  185. package/src/skills/index.ts +36 -0
  186. package/src/skills/loader.test.ts +300 -0
  187. package/src/skills/loader.ts +423 -0
  188. package/src/skills/remote.test.ts +704 -0
  189. package/src/skills/remote.ts +604 -0
  190. package/src/triggers/matcher.test.ts +277 -0
  191. package/src/triggers/matcher.ts +152 -0
  192. package/src/types/index.ts +194 -0
  193. package/src/utils/async.ts +18 -0
  194. package/src/utils/index.test.ts +84 -0
  195. package/src/utils/index.ts +50 -0
  196. package/tsconfig.json +25 -0
  197. package/vitest.config.ts +8 -0
  198. package/vitest.integration.config.ts +11 -0
  199. 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
+ }