docrev 0.9.13 → 0.9.14
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/.claude/settings.local.json +9 -9
- package/.gitattributes +1 -1
- package/CHANGELOG.md +149 -149
- package/PLAN-tables-and-postprocess.md +850 -850
- package/README.md +391 -391
- package/bin/rev.js +11 -11
- package/bin/rev.ts +145 -145
- package/completions/rev.bash +127 -127
- package/completions/rev.ps1 +210 -210
- package/completions/rev.zsh +207 -207
- package/dev_notes/stress2/build_adversarial.ts +186 -186
- package/dev_notes/stress2/drift_matcher.ts +62 -62
- package/dev_notes/stress2/probe_anchors.ts +35 -35
- package/dev_notes/stress2/project/discussion.before.md +3 -3
- package/dev_notes/stress2/project/discussion.md +3 -3
- package/dev_notes/stress2/project/methods.before.md +20 -20
- package/dev_notes/stress2/project/methods.md +20 -20
- package/dev_notes/stress2/project/rev.yaml +5 -5
- package/dev_notes/stress2/project/sections.yaml +4 -4
- package/dev_notes/stress2/sections.yaml +5 -5
- package/dev_notes/stress2/trace_placement.ts +50 -50
- package/dev_notes/stresstest_boundaries.ts +27 -27
- package/dev_notes/stresstest_drift_apply.ts +43 -43
- package/dev_notes/stresstest_drift_compare.ts +43 -43
- package/dev_notes/stresstest_drift_v2.ts +54 -54
- package/dev_notes/stresstest_inspect.ts +54 -54
- package/dev_notes/stresstest_pstyle.ts +55 -55
- package/dev_notes/stresstest_section_debug.ts +23 -23
- package/dev_notes/stresstest_split.ts +70 -70
- package/dev_notes/stresstest_trace.ts +19 -19
- package/dev_notes/stresstest_verify_no_overwrite.ts +40 -40
- package/dist/lib/build.d.ts +38 -1
- package/dist/lib/build.d.ts.map +1 -1
- package/dist/lib/build.js +68 -30
- package/dist/lib/build.js.map +1 -1
- package/dist/lib/commands/build.d.ts.map +1 -1
- package/dist/lib/commands/build.js +38 -5
- package/dist/lib/commands/build.js.map +1 -1
- package/dist/lib/commands/utilities.js +164 -164
- package/dist/lib/commands/word-tools.js +8 -8
- package/dist/lib/grammar.js +3 -3
- package/dist/lib/pdf-comments.js +44 -44
- package/dist/lib/plugins.js +57 -57
- package/dist/lib/pptx-themes.js +115 -115
- package/dist/lib/spelling.js +2 -2
- package/dist/lib/templates.js +387 -387
- package/dist/lib/themes.js +51 -51
- package/eslint.config.js +27 -27
- package/lib/anchor-match.ts +276 -276
- package/lib/annotations.ts +644 -644
- package/lib/build.ts +1300 -1251
- package/lib/citations.ts +160 -160
- package/lib/commands/build.ts +833 -801
- package/lib/commands/citations.ts +515 -515
- package/lib/commands/comments.ts +1050 -1050
- package/lib/commands/context.ts +174 -174
- package/lib/commands/core.ts +309 -309
- package/lib/commands/doi.ts +435 -435
- package/lib/commands/file-ops.ts +372 -372
- package/lib/commands/history.ts +320 -320
- package/lib/commands/index.ts +87 -87
- package/lib/commands/init.ts +259 -259
- package/lib/commands/merge-resolve.ts +378 -378
- package/lib/commands/preview.ts +178 -178
- package/lib/commands/project-info.ts +244 -244
- package/lib/commands/quality.ts +517 -517
- package/lib/commands/response.ts +454 -454
- package/lib/commands/section-boundaries.ts +82 -82
- package/lib/commands/sections.ts +451 -451
- package/lib/commands/sync.ts +706 -706
- package/lib/commands/text-ops.ts +449 -449
- package/lib/commands/utilities.ts +448 -448
- package/lib/commands/verify-anchors.ts +272 -272
- package/lib/commands/word-tools.ts +340 -340
- package/lib/comment-realign.ts +517 -517
- package/lib/config.ts +84 -84
- package/lib/crossref.ts +781 -781
- package/lib/csl.ts +191 -191
- package/lib/dependencies.ts +98 -98
- package/lib/diff-engine.ts +465 -465
- package/lib/doi-cache.ts +115 -115
- package/lib/doi.ts +897 -897
- package/lib/equations.ts +506 -506
- package/lib/errors.ts +346 -346
- package/lib/format.ts +541 -541
- package/lib/git.ts +326 -326
- package/lib/grammar.ts +303 -303
- package/lib/image-registry.ts +180 -180
- package/lib/import.ts +911 -911
- package/lib/journals.ts +543 -543
- package/lib/merge.ts +633 -633
- package/lib/orcid.ts +144 -144
- package/lib/pdf-comments.ts +263 -263
- package/lib/pdf-import.ts +524 -524
- package/lib/plugins.ts +362 -362
- package/lib/postprocess.ts +188 -188
- package/lib/pptx-color-filter.lua +37 -37
- package/lib/pptx-template.ts +469 -469
- package/lib/pptx-themes.ts +483 -483
- package/lib/protect-restore.ts +520 -520
- package/lib/rate-limiter.ts +94 -94
- package/lib/response.ts +197 -197
- package/lib/restore-references.ts +240 -240
- package/lib/review.ts +327 -327
- package/lib/schema.ts +417 -417
- package/lib/scientific-words.ts +73 -73
- package/lib/sections.ts +335 -335
- package/lib/slides.ts +756 -756
- package/lib/spelling.ts +334 -334
- package/lib/templates.ts +526 -526
- package/lib/themes.ts +742 -742
- package/lib/trackchanges.ts +247 -247
- package/lib/tui.ts +450 -450
- package/lib/types.ts +550 -550
- package/lib/undo.ts +250 -250
- package/lib/utils.ts +69 -69
- package/lib/variables.ts +179 -179
- package/lib/word-extraction.ts +806 -806
- package/lib/word.ts +643 -643
- package/lib/wordcomments.ts +817 -817
- package/package.json +137 -137
- package/scripts/postbuild.js +28 -28
- package/skill/REFERENCE.md +431 -431
- package/skill/SKILL.md +258 -258
- package/tsconfig.json +26 -26
- package/types/index.d.ts +525 -525
package/lib/commands/history.ts
CHANGED
|
@@ -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
|
+
}
|