docrev 0.9.13 → 0.9.15

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 (126) hide show
  1. package/.claude/settings.local.json +9 -9
  2. package/.gitattributes +1 -1
  3. package/CHANGELOG.md +149 -149
  4. package/PLAN-tables-and-postprocess.md +850 -850
  5. package/README.md +411 -391
  6. package/bin/rev.js +11 -11
  7. package/bin/rev.ts +145 -145
  8. package/completions/rev.bash +127 -127
  9. package/completions/rev.ps1 +210 -210
  10. package/completions/rev.zsh +207 -207
  11. package/dev_notes/stress2/build_adversarial.ts +186 -186
  12. package/dev_notes/stress2/drift_matcher.ts +62 -62
  13. package/dev_notes/stress2/probe_anchors.ts +35 -35
  14. package/dev_notes/stress2/project/discussion.before.md +3 -3
  15. package/dev_notes/stress2/project/discussion.md +3 -3
  16. package/dev_notes/stress2/project/methods.before.md +20 -20
  17. package/dev_notes/stress2/project/methods.md +20 -20
  18. package/dev_notes/stress2/project/rev.yaml +5 -5
  19. package/dev_notes/stress2/project/sections.yaml +4 -4
  20. package/dev_notes/stress2/sections.yaml +5 -5
  21. package/dev_notes/stress2/trace_placement.ts +50 -50
  22. package/dev_notes/stresstest_boundaries.ts +27 -27
  23. package/dev_notes/stresstest_drift_apply.ts +43 -43
  24. package/dev_notes/stresstest_drift_compare.ts +43 -43
  25. package/dev_notes/stresstest_drift_v2.ts +54 -54
  26. package/dev_notes/stresstest_inspect.ts +54 -54
  27. package/dev_notes/stresstest_pstyle.ts +55 -55
  28. package/dev_notes/stresstest_section_debug.ts +23 -23
  29. package/dev_notes/stresstest_split.ts +70 -70
  30. package/dev_notes/stresstest_trace.ts +19 -19
  31. package/dev_notes/stresstest_verify_no_overwrite.ts +40 -40
  32. package/dist/lib/build.d.ts +38 -1
  33. package/dist/lib/build.d.ts.map +1 -1
  34. package/dist/lib/build.js +68 -30
  35. package/dist/lib/build.js.map +1 -1
  36. package/dist/lib/commands/build.d.ts.map +1 -1
  37. package/dist/lib/commands/build.js +38 -5
  38. package/dist/lib/commands/build.js.map +1 -1
  39. package/dist/lib/commands/utilities.js +164 -164
  40. package/dist/lib/commands/word-tools.js +8 -8
  41. package/dist/lib/grammar.js +3 -3
  42. package/dist/lib/pdf-comments.js +44 -44
  43. package/dist/lib/plugins.js +57 -57
  44. package/dist/lib/pptx-themes.js +115 -115
  45. package/dist/lib/spelling.js +2 -2
  46. package/dist/lib/templates.js +387 -387
  47. package/dist/lib/themes.js +51 -51
  48. package/eslint.config.js +27 -27
  49. package/lib/anchor-match.ts +276 -276
  50. package/lib/annotations.ts +644 -644
  51. package/lib/build.ts +1300 -1251
  52. package/lib/citations.ts +160 -160
  53. package/lib/commands/build.ts +833 -801
  54. package/lib/commands/citations.ts +515 -515
  55. package/lib/commands/comments.ts +1050 -1050
  56. package/lib/commands/context.ts +174 -174
  57. package/lib/commands/core.ts +309 -309
  58. package/lib/commands/doi.ts +435 -435
  59. package/lib/commands/file-ops.ts +372 -372
  60. package/lib/commands/history.ts +320 -320
  61. package/lib/commands/index.ts +87 -87
  62. package/lib/commands/init.ts +259 -259
  63. package/lib/commands/merge-resolve.ts +378 -378
  64. package/lib/commands/preview.ts +178 -178
  65. package/lib/commands/project-info.ts +244 -244
  66. package/lib/commands/quality.ts +517 -517
  67. package/lib/commands/response.ts +454 -454
  68. package/lib/commands/section-boundaries.ts +82 -82
  69. package/lib/commands/sections.ts +451 -451
  70. package/lib/commands/sync.ts +706 -706
  71. package/lib/commands/text-ops.ts +449 -449
  72. package/lib/commands/utilities.ts +448 -448
  73. package/lib/commands/verify-anchors.ts +272 -272
  74. package/lib/commands/word-tools.ts +340 -340
  75. package/lib/comment-realign.ts +517 -517
  76. package/lib/config.ts +84 -84
  77. package/lib/crossref.ts +781 -781
  78. package/lib/csl.ts +191 -191
  79. package/lib/dependencies.ts +98 -98
  80. package/lib/diff-engine.ts +465 -465
  81. package/lib/doi-cache.ts +115 -115
  82. package/lib/doi.ts +897 -897
  83. package/lib/equations.ts +506 -506
  84. package/lib/errors.ts +346 -346
  85. package/lib/format.ts +541 -541
  86. package/lib/git.ts +326 -326
  87. package/lib/grammar.ts +303 -303
  88. package/lib/image-registry.ts +180 -180
  89. package/lib/import.ts +911 -911
  90. package/lib/journals.ts +543 -543
  91. package/lib/merge.ts +633 -633
  92. package/lib/orcid.ts +144 -144
  93. package/lib/pdf-comments.ts +263 -263
  94. package/lib/pdf-import.ts +524 -524
  95. package/lib/plugins.ts +362 -362
  96. package/lib/postprocess.ts +188 -188
  97. package/lib/pptx-color-filter.lua +37 -37
  98. package/lib/pptx-template.ts +469 -469
  99. package/lib/pptx-themes.ts +483 -483
  100. package/lib/protect-restore.ts +520 -520
  101. package/lib/rate-limiter.ts +94 -94
  102. package/lib/response.ts +197 -197
  103. package/lib/restore-references.ts +240 -240
  104. package/lib/review.ts +327 -327
  105. package/lib/schema.ts +417 -417
  106. package/lib/scientific-words.ts +73 -73
  107. package/lib/sections.ts +335 -335
  108. package/lib/slides.ts +756 -756
  109. package/lib/spelling.ts +334 -334
  110. package/lib/templates.ts +526 -526
  111. package/lib/themes.ts +742 -742
  112. package/lib/trackchanges.ts +247 -247
  113. package/lib/tui.ts +450 -450
  114. package/lib/types.ts +550 -550
  115. package/lib/undo.ts +250 -250
  116. package/lib/utils.ts +69 -69
  117. package/lib/variables.ts +179 -179
  118. package/lib/word-extraction.ts +806 -806
  119. package/lib/word.ts +643 -643
  120. package/lib/wordcomments.ts +817 -817
  121. package/package.json +137 -137
  122. package/scripts/postbuild.js +28 -28
  123. package/skill/REFERENCE.md +473 -431
  124. package/skill/SKILL.md +274 -258
  125. package/tsconfig.json +26 -26
  126. package/types/index.d.ts +525 -525
@@ -1,320 +1,320 @@
1
- /**
2
- * History commands: diff, history, contributors
3
- *
4
- * Commands for git-based revision tracking and author statistics.
5
- */
6
-
7
- import type { Command } from 'commander';
8
- import {
9
- chalk,
10
- fs,
11
- path,
12
- fmt,
13
- loadBuildConfig,
14
- } from './context.js';
15
-
16
- interface DiffOptions {
17
- files?: string;
18
- stat?: boolean;
19
- }
20
-
21
- interface HistoryOptions {
22
- limit: string;
23
- }
24
-
25
- interface ContributorsOptions {
26
- blame?: boolean;
27
- }
28
-
29
- /**
30
- * Register history commands with the program
31
- */
32
- export function register(program: Command): void {
33
- // ==========================================================================
34
- // DIFF command - Compare sections against git history
35
- // ==========================================================================
36
-
37
- program
38
- .command('diff')
39
- .description('Compare sections against git history')
40
- .argument('[ref]', 'Git reference to compare against (default: main/master)')
41
- .option('-f, --files <files>', 'Specific files to compare (comma-separated)')
42
- .option('--stat', 'Show only statistics, not full diff')
43
- .action(async (ref: string | undefined, options: DiffOptions) => {
44
- const {
45
- isGitRepo,
46
- getDefaultBranch,
47
- getCurrentBranch,
48
- getChangedFiles,
49
- getWordCountDiff,
50
- compareFileVersions,
51
- } = await import('../git.js');
52
-
53
- if (!isGitRepo()) {
54
- console.error(fmt.status('error', 'Not a git repository'));
55
- process.exit(1);
56
- }
57
-
58
- const compareRef = ref || getDefaultBranch();
59
- const currentBranch = getCurrentBranch();
60
-
61
- console.log(fmt.header('Git Diff'));
62
- console.log(chalk.dim(` Comparing: ${compareRef} → ${currentBranch || 'HEAD'}`));
63
- console.log();
64
-
65
- // Get files to compare
66
- let filesToCompare: string[];
67
- if (options.files) {
68
- filesToCompare = options.files.split(',').map(f => f.trim());
69
- } else {
70
- // Default to markdown section files
71
- filesToCompare = fs.readdirSync('.').filter(f =>
72
- f.endsWith('.md') && !['README.md', 'CLAUDE.md'].includes(f)
73
- );
74
- }
75
-
76
- if (filesToCompare.length === 0) {
77
- console.log(fmt.status('info', 'No markdown files found'));
78
- return;
79
- }
80
-
81
- // Get changed files from git
82
- const changedFiles = getChangedFiles(compareRef);
83
- const changedSet = new Set(changedFiles.map(f => f.file));
84
-
85
- // Get word count differences
86
- const { total, byFile } = getWordCountDiff(filesToCompare, compareRef);
87
-
88
- // Show results
89
- const rows: string[][] = [];
90
- for (const file of filesToCompare) {
91
- const stats = byFile[file];
92
- if (stats && (stats.added > 0 || stats.removed > 0)) {
93
- const status = changedSet.has(file)
94
- ? changedFiles.find(f => f.file === file)?.status || 'modified'
95
- : 'unchanged';
96
- rows.push([
97
- file,
98
- status,
99
- chalk.green(`+${stats.added}`),
100
- chalk.red(`-${stats.removed}`),
101
- ]);
102
- }
103
- }
104
-
105
- if (rows.length === 0) {
106
- console.log(fmt.status('success', 'No changes detected'));
107
- return;
108
- }
109
-
110
- console.log(fmt.table(['File', 'Status', 'Added', 'Removed'], rows));
111
- console.log();
112
- console.log(chalk.dim(`Total: ${chalk.green(`+${total.added}`)} words, ${chalk.red(`-${total.removed}`)} words`));
113
-
114
- // Show detailed diff if not --stat
115
- if (!options.stat && rows.length > 0) {
116
- console.log();
117
- console.log(chalk.cyan('Changed sections:'));
118
- for (const file of filesToCompare) {
119
- const stats = byFile[file];
120
- if (stats && (stats.added > 0 || stats.removed > 0)) {
121
- const { changes } = compareFileVersions(file, compareRef);
122
- console.log(chalk.bold(`\n ${file}:`));
123
-
124
- // Show first few significant changes
125
- let shown = 0;
126
- for (const change of changes) {
127
- if (shown >= 3) {
128
- console.log(chalk.dim(' ...'));
129
- break;
130
- }
131
- const preview = change.value.slice(0, 60).replace(/\n/g, ' ');
132
- if (change.added) {
133
- console.log(chalk.green(` + "${preview}..."`));
134
- } else if (change.removed) {
135
- console.log(chalk.red(` - "${preview}..."`));
136
- }
137
- shown++;
138
- }
139
- }
140
- }
141
- }
142
- });
143
-
144
- // ==========================================================================
145
- // HISTORY command - Show revision history
146
- // ==========================================================================
147
-
148
- program
149
- .command('history')
150
- .description('Show revision history for section files')
151
- .argument('[file]', 'Specific file (default: all sections)')
152
- .option('-n, --limit <count>', 'Number of commits to show', '10')
153
- .action(async (file: string | undefined, options: HistoryOptions) => {
154
- const {
155
- isGitRepo,
156
- getFileHistory,
157
- getRecentCommits,
158
- hasUncommittedChanges,
159
- } = await import('../git.js');
160
-
161
- if (!isGitRepo()) {
162
- console.error(fmt.status('error', 'Not a git repository'));
163
- process.exit(1);
164
- }
165
-
166
- const limit = parseInt(options.limit) || 10;
167
-
168
- console.log(fmt.header('Revision History'));
169
- console.log();
170
-
171
- if (file) {
172
- // Show history for specific file
173
- if (!fs.existsSync(file)) {
174
- console.error(fmt.status('error', `File not found: ${file}`));
175
- process.exit(1);
176
- }
177
-
178
- const history = getFileHistory(file, limit);
179
-
180
- if (history.length === 0) {
181
- console.log(fmt.status('info', 'No history found (file may not be committed)'));
182
- return;
183
- }
184
-
185
- console.log(chalk.cyan(`History for ${file}:`));
186
- console.log();
187
-
188
- for (const commit of history) {
189
- const date = new Date(commit.date).toLocaleDateString();
190
- console.log(` ${chalk.yellow(commit.hash)} ${chalk.dim(date)}`);
191
- console.log(` ${commit.message}`);
192
- }
193
- } else {
194
- // Show recent commits affecting any file
195
- const commits = getRecentCommits(limit);
196
-
197
- if (commits.length === 0) {
198
- console.log(fmt.status('info', 'No commits found'));
199
- return;
200
- }
201
-
202
- if (hasUncommittedChanges()) {
203
- console.log(chalk.yellow(' * Uncommitted changes'));
204
- console.log();
205
- }
206
-
207
- for (const commit of commits) {
208
- const date = new Date(commit.date).toLocaleDateString();
209
- console.log(` ${chalk.yellow(commit.hash)} ${chalk.dim(date)} ${chalk.blue(commit.author)}`);
210
- console.log(` ${commit.message}`);
211
- }
212
- }
213
- });
214
-
215
- // ==========================================================================
216
- // CONTRIBUTORS command - Show who wrote what
217
- // ==========================================================================
218
-
219
- program
220
- .command('contributors')
221
- .alias('authors')
222
- .description('Show author contributions across section files')
223
- .argument('[file]', 'Specific file (default: all sections)')
224
- .option('--blame', 'Show detailed line-by-line blame for a file')
225
- .action(async (file: string | undefined, options: ContributorsOptions) => {
226
- const { isGitRepo, getAuthorStats, getContributors, getFileBlame } = await import('../git.js');
227
-
228
- if (!isGitRepo()) {
229
- console.error(fmt.status('error', 'Not a git repository'));
230
- process.exit(1);
231
- }
232
-
233
- console.log(fmt.header('Contributors'));
234
- console.log();
235
-
236
- if (file) {
237
- // Show stats for specific file
238
- if (!fs.existsSync(file)) {
239
- console.error(fmt.status('error', `File not found: ${file}`));
240
- process.exit(1);
241
- }
242
-
243
- if (options.blame) {
244
- // Detailed blame output
245
- const blame = getFileBlame(file);
246
- if (blame.length === 0) {
247
- console.log(fmt.status('info', 'No git history (file may not be committed)'));
248
- return;
249
- }
250
-
251
- console.log(chalk.cyan(`Blame for ${file}:`));
252
- console.log();
253
-
254
- for (const entry of blame) {
255
- const authorShort = entry.author.slice(0, 15).padEnd(15);
256
- const content = entry.content.length > 60 ? entry.content.slice(0, 60) + '...' : entry.content;
257
- console.log(` ${chalk.dim(entry.hash)} ${chalk.blue(authorShort)} ${chalk.dim(`L${String(entry.line).padStart(3)}`)} ${content}`);
258
- }
259
- } else {
260
- // Summary stats
261
- const stats = getAuthorStats(file);
262
- if (Object.keys(stats).length === 0) {
263
- console.log(fmt.status('info', 'No git history (file may not be committed)'));
264
- return;
265
- }
266
-
267
- console.log(chalk.cyan(`Authors for ${file}:`));
268
- console.log();
269
-
270
- const sorted = Object.entries(stats).sort((a, b) => b[1].lines - a[1].lines);
271
- for (const [author, data] of sorted) {
272
- const bar = '█'.repeat(Math.ceil(data.percentage / 5));
273
- console.log(` ${chalk.blue(author.padEnd(25))} ${chalk.dim(String(data.lines).padStart(4))} lines ${chalk.green(bar)} ${data.percentage}%`);
274
- }
275
- }
276
- } else {
277
- // Show contributors across all sections
278
- let config: { sections?: string[] } = {};
279
- try {
280
- config = loadBuildConfig(process.cwd()) || {};
281
- } catch {
282
- // Not in a rev project
283
- }
284
-
285
- let sections = config.sections || [];
286
- if (sections.length === 0) {
287
- sections = fs.readdirSync('.').filter(f =>
288
- f.endsWith('.md') && !['README.md', 'CLAUDE.md', 'paper.md'].includes(f)
289
- );
290
- }
291
-
292
- if (sections.length === 0) {
293
- console.error(fmt.status('error', 'No section files found'));
294
- process.exit(1);
295
- }
296
-
297
- const contributors = getContributors(sections);
298
-
299
- if (Object.keys(contributors).length === 0) {
300
- console.log(fmt.status('info', 'No git history found'));
301
- return;
302
- }
303
-
304
- const sorted = Object.entries(contributors).sort((a, b) => b[1].lines - a[1].lines);
305
- const totalLines = sorted.reduce((sum, [, data]) => sum + data.lines, 0);
306
-
307
- console.log(chalk.cyan('Project contributors:'));
308
- console.log();
309
-
310
- for (const [author, data] of sorted) {
311
- const pct = Math.round((data.lines / totalLines) * 100);
312
- const bar = '█'.repeat(Math.ceil(pct / 5));
313
- console.log(` ${chalk.blue(author.padEnd(25))} ${chalk.dim(String(data.lines).padStart(5))} lines ${chalk.dim(String(data.files))} files ${chalk.green(bar)} ${pct}%`);
314
- }
315
-
316
- console.log();
317
- console.log(chalk.dim(` Total: ${totalLines} lines across ${sections.length} files`));
318
- }
319
- });
320
- }
1
+ /**
2
+ * History commands: diff, history, contributors
3
+ *
4
+ * Commands for git-based revision tracking and author statistics.
5
+ */
6
+
7
+ import type { Command } from 'commander';
8
+ import {
9
+ chalk,
10
+ fs,
11
+ path,
12
+ fmt,
13
+ loadBuildConfig,
14
+ } from './context.js';
15
+
16
+ interface DiffOptions {
17
+ files?: string;
18
+ stat?: boolean;
19
+ }
20
+
21
+ interface HistoryOptions {
22
+ limit: string;
23
+ }
24
+
25
+ interface ContributorsOptions {
26
+ blame?: boolean;
27
+ }
28
+
29
+ /**
30
+ * Register history commands with the program
31
+ */
32
+ export function register(program: Command): void {
33
+ // ==========================================================================
34
+ // DIFF command - Compare sections against git history
35
+ // ==========================================================================
36
+
37
+ program
38
+ .command('diff')
39
+ .description('Compare sections against git history')
40
+ .argument('[ref]', 'Git reference to compare against (default: main/master)')
41
+ .option('-f, --files <files>', 'Specific files to compare (comma-separated)')
42
+ .option('--stat', 'Show only statistics, not full diff')
43
+ .action(async (ref: string | undefined, options: DiffOptions) => {
44
+ const {
45
+ isGitRepo,
46
+ getDefaultBranch,
47
+ getCurrentBranch,
48
+ getChangedFiles,
49
+ getWordCountDiff,
50
+ compareFileVersions,
51
+ } = await import('../git.js');
52
+
53
+ if (!isGitRepo()) {
54
+ console.error(fmt.status('error', 'Not a git repository'));
55
+ process.exit(1);
56
+ }
57
+
58
+ const compareRef = ref || getDefaultBranch();
59
+ const currentBranch = getCurrentBranch();
60
+
61
+ console.log(fmt.header('Git Diff'));
62
+ console.log(chalk.dim(` Comparing: ${compareRef} → ${currentBranch || 'HEAD'}`));
63
+ console.log();
64
+
65
+ // Get files to compare
66
+ let filesToCompare: string[];
67
+ if (options.files) {
68
+ filesToCompare = options.files.split(',').map(f => f.trim());
69
+ } else {
70
+ // Default to markdown section files
71
+ filesToCompare = fs.readdirSync('.').filter(f =>
72
+ f.endsWith('.md') && !['README.md', 'CLAUDE.md'].includes(f)
73
+ );
74
+ }
75
+
76
+ if (filesToCompare.length === 0) {
77
+ console.log(fmt.status('info', 'No markdown files found'));
78
+ return;
79
+ }
80
+
81
+ // Get changed files from git
82
+ const changedFiles = getChangedFiles(compareRef);
83
+ const changedSet = new Set(changedFiles.map(f => f.file));
84
+
85
+ // Get word count differences
86
+ const { total, byFile } = getWordCountDiff(filesToCompare, compareRef);
87
+
88
+ // Show results
89
+ const rows: string[][] = [];
90
+ for (const file of filesToCompare) {
91
+ const stats = byFile[file];
92
+ if (stats && (stats.added > 0 || stats.removed > 0)) {
93
+ const status = changedSet.has(file)
94
+ ? changedFiles.find(f => f.file === file)?.status || 'modified'
95
+ : 'unchanged';
96
+ rows.push([
97
+ file,
98
+ status,
99
+ chalk.green(`+${stats.added}`),
100
+ chalk.red(`-${stats.removed}`),
101
+ ]);
102
+ }
103
+ }
104
+
105
+ if (rows.length === 0) {
106
+ console.log(fmt.status('success', 'No changes detected'));
107
+ return;
108
+ }
109
+
110
+ console.log(fmt.table(['File', 'Status', 'Added', 'Removed'], rows));
111
+ console.log();
112
+ console.log(chalk.dim(`Total: ${chalk.green(`+${total.added}`)} words, ${chalk.red(`-${total.removed}`)} words`));
113
+
114
+ // Show detailed diff if not --stat
115
+ if (!options.stat && rows.length > 0) {
116
+ console.log();
117
+ console.log(chalk.cyan('Changed sections:'));
118
+ for (const file of filesToCompare) {
119
+ const stats = byFile[file];
120
+ if (stats && (stats.added > 0 || stats.removed > 0)) {
121
+ const { changes } = compareFileVersions(file, compareRef);
122
+ console.log(chalk.bold(`\n ${file}:`));
123
+
124
+ // Show first few significant changes
125
+ let shown = 0;
126
+ for (const change of changes) {
127
+ if (shown >= 3) {
128
+ console.log(chalk.dim(' ...'));
129
+ break;
130
+ }
131
+ const preview = change.value.slice(0, 60).replace(/\n/g, ' ');
132
+ if (change.added) {
133
+ console.log(chalk.green(` + "${preview}..."`));
134
+ } else if (change.removed) {
135
+ console.log(chalk.red(` - "${preview}..."`));
136
+ }
137
+ shown++;
138
+ }
139
+ }
140
+ }
141
+ }
142
+ });
143
+
144
+ // ==========================================================================
145
+ // HISTORY command - Show revision history
146
+ // ==========================================================================
147
+
148
+ program
149
+ .command('history')
150
+ .description('Show revision history for section files')
151
+ .argument('[file]', 'Specific file (default: all sections)')
152
+ .option('-n, --limit <count>', 'Number of commits to show', '10')
153
+ .action(async (file: string | undefined, options: HistoryOptions) => {
154
+ const {
155
+ isGitRepo,
156
+ getFileHistory,
157
+ getRecentCommits,
158
+ hasUncommittedChanges,
159
+ } = await import('../git.js');
160
+
161
+ if (!isGitRepo()) {
162
+ console.error(fmt.status('error', 'Not a git repository'));
163
+ process.exit(1);
164
+ }
165
+
166
+ const limit = parseInt(options.limit) || 10;
167
+
168
+ console.log(fmt.header('Revision History'));
169
+ console.log();
170
+
171
+ if (file) {
172
+ // Show history for specific file
173
+ if (!fs.existsSync(file)) {
174
+ console.error(fmt.status('error', `File not found: ${file}`));
175
+ process.exit(1);
176
+ }
177
+
178
+ const history = getFileHistory(file, limit);
179
+
180
+ if (history.length === 0) {
181
+ console.log(fmt.status('info', 'No history found (file may not be committed)'));
182
+ return;
183
+ }
184
+
185
+ console.log(chalk.cyan(`History for ${file}:`));
186
+ console.log();
187
+
188
+ for (const commit of history) {
189
+ const date = new Date(commit.date).toLocaleDateString();
190
+ console.log(` ${chalk.yellow(commit.hash)} ${chalk.dim(date)}`);
191
+ console.log(` ${commit.message}`);
192
+ }
193
+ } else {
194
+ // Show recent commits affecting any file
195
+ const commits = getRecentCommits(limit);
196
+
197
+ if (commits.length === 0) {
198
+ console.log(fmt.status('info', 'No commits found'));
199
+ return;
200
+ }
201
+
202
+ if (hasUncommittedChanges()) {
203
+ console.log(chalk.yellow(' * Uncommitted changes'));
204
+ console.log();
205
+ }
206
+
207
+ for (const commit of commits) {
208
+ const date = new Date(commit.date).toLocaleDateString();
209
+ console.log(` ${chalk.yellow(commit.hash)} ${chalk.dim(date)} ${chalk.blue(commit.author)}`);
210
+ console.log(` ${commit.message}`);
211
+ }
212
+ }
213
+ });
214
+
215
+ // ==========================================================================
216
+ // CONTRIBUTORS command - Show who wrote what
217
+ // ==========================================================================
218
+
219
+ program
220
+ .command('contributors')
221
+ .alias('authors')
222
+ .description('Show author contributions across section files')
223
+ .argument('[file]', 'Specific file (default: all sections)')
224
+ .option('--blame', 'Show detailed line-by-line blame for a file')
225
+ .action(async (file: string | undefined, options: ContributorsOptions) => {
226
+ const { isGitRepo, getAuthorStats, getContributors, getFileBlame } = await import('../git.js');
227
+
228
+ if (!isGitRepo()) {
229
+ console.error(fmt.status('error', 'Not a git repository'));
230
+ process.exit(1);
231
+ }
232
+
233
+ console.log(fmt.header('Contributors'));
234
+ console.log();
235
+
236
+ if (file) {
237
+ // Show stats for specific file
238
+ if (!fs.existsSync(file)) {
239
+ console.error(fmt.status('error', `File not found: ${file}`));
240
+ process.exit(1);
241
+ }
242
+
243
+ if (options.blame) {
244
+ // Detailed blame output
245
+ const blame = getFileBlame(file);
246
+ if (blame.length === 0) {
247
+ console.log(fmt.status('info', 'No git history (file may not be committed)'));
248
+ return;
249
+ }
250
+
251
+ console.log(chalk.cyan(`Blame for ${file}:`));
252
+ console.log();
253
+
254
+ for (const entry of blame) {
255
+ const authorShort = entry.author.slice(0, 15).padEnd(15);
256
+ const content = entry.content.length > 60 ? entry.content.slice(0, 60) + '...' : entry.content;
257
+ console.log(` ${chalk.dim(entry.hash)} ${chalk.blue(authorShort)} ${chalk.dim(`L${String(entry.line).padStart(3)}`)} ${content}`);
258
+ }
259
+ } else {
260
+ // Summary stats
261
+ const stats = getAuthorStats(file);
262
+ if (Object.keys(stats).length === 0) {
263
+ console.log(fmt.status('info', 'No git history (file may not be committed)'));
264
+ return;
265
+ }
266
+
267
+ console.log(chalk.cyan(`Authors for ${file}:`));
268
+ console.log();
269
+
270
+ const sorted = Object.entries(stats).sort((a, b) => b[1].lines - a[1].lines);
271
+ for (const [author, data] of sorted) {
272
+ const bar = '█'.repeat(Math.ceil(data.percentage / 5));
273
+ console.log(` ${chalk.blue(author.padEnd(25))} ${chalk.dim(String(data.lines).padStart(4))} lines ${chalk.green(bar)} ${data.percentage}%`);
274
+ }
275
+ }
276
+ } else {
277
+ // Show contributors across all sections
278
+ let config: { sections?: string[] } = {};
279
+ try {
280
+ config = loadBuildConfig(process.cwd()) || {};
281
+ } catch {
282
+ // Not in a rev project
283
+ }
284
+
285
+ let sections = config.sections || [];
286
+ if (sections.length === 0) {
287
+ sections = fs.readdirSync('.').filter(f =>
288
+ f.endsWith('.md') && !['README.md', 'CLAUDE.md', 'paper.md'].includes(f)
289
+ );
290
+ }
291
+
292
+ if (sections.length === 0) {
293
+ console.error(fmt.status('error', 'No section files found'));
294
+ process.exit(1);
295
+ }
296
+
297
+ const contributors = getContributors(sections);
298
+
299
+ if (Object.keys(contributors).length === 0) {
300
+ console.log(fmt.status('info', 'No git history found'));
301
+ return;
302
+ }
303
+
304
+ const sorted = Object.entries(contributors).sort((a, b) => b[1].lines - a[1].lines);
305
+ const totalLines = sorted.reduce((sum, [, data]) => sum + data.lines, 0);
306
+
307
+ console.log(chalk.cyan('Project contributors:'));
308
+ console.log();
309
+
310
+ for (const [author, data] of sorted) {
311
+ const pct = Math.round((data.lines / totalLines) * 100);
312
+ const bar = '█'.repeat(Math.ceil(pct / 5));
313
+ console.log(` ${chalk.blue(author.padEnd(25))} ${chalk.dim(String(data.lines).padStart(5))} lines ${chalk.dim(String(data.files))} files ${chalk.green(bar)} ${pct}%`);
314
+ }
315
+
316
+ console.log();
317
+ console.log(chalk.dim(` Total: ${totalLines} lines across ${sections.length} files`));
318
+ }
319
+ });
320
+ }