docrev 0.10.0 → 0.10.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (126) hide show
  1. package/.gitattributes +1 -1
  2. package/CHANGELOG.md +173 -164
  3. package/PLAN-tables-and-postprocess.md +850 -850
  4. package/README.md +431 -431
  5. package/bin/rev.js +11 -11
  6. package/bin/rev.ts +145 -145
  7. package/completions/rev.bash +127 -127
  8. package/completions/rev.ps1 +210 -210
  9. package/completions/rev.zsh +207 -207
  10. package/dist/lib/anchor-match.d.ts +1 -1
  11. package/dist/lib/anchor-match.d.ts.map +1 -1
  12. package/dist/lib/anchor-match.js +17 -47
  13. package/dist/lib/anchor-match.js.map +1 -1
  14. package/dist/lib/build.js +4 -4
  15. package/dist/lib/commands/context.d.ts +1 -1
  16. package/dist/lib/commands/context.d.ts.map +1 -1
  17. package/dist/lib/commands/context.js +1 -1
  18. package/dist/lib/commands/context.js.map +1 -1
  19. package/dist/lib/commands/sections.js +7 -7
  20. package/dist/lib/commands/sections.js.map +1 -1
  21. package/dist/lib/commands/sync.d.ts.map +1 -1
  22. package/dist/lib/commands/sync.js +15 -14
  23. package/dist/lib/commands/sync.js.map +1 -1
  24. package/dist/lib/commands/utilities.js +164 -164
  25. package/dist/lib/commands/verify-anchors.js +6 -6
  26. package/dist/lib/commands/verify-anchors.js.map +1 -1
  27. package/dist/lib/commands/word-tools.js +8 -8
  28. package/dist/lib/grammar.js +3 -3
  29. package/dist/lib/macro-filter.lua +201 -201
  30. package/dist/lib/pdf-comments.js +44 -44
  31. package/dist/lib/plugins.js +57 -57
  32. package/dist/lib/pptx-color-filter.lua +37 -37
  33. package/dist/lib/pptx-themes.js +115 -115
  34. package/dist/lib/sections.d.ts +35 -0
  35. package/dist/lib/sections.d.ts.map +1 -1
  36. package/dist/lib/sections.js +81 -0
  37. package/dist/lib/sections.js.map +1 -1
  38. package/dist/lib/spelling.js +2 -2
  39. package/dist/lib/templates.js +387 -387
  40. package/dist/lib/themes.js +51 -51
  41. package/docs-src/build.py +113 -113
  42. package/docs-src/extra.css +208 -208
  43. package/docs-src/md-to-html.lua +6 -6
  44. package/docs-src/template.html +116 -116
  45. package/eslint.config.js +27 -27
  46. package/lib/anchor-match.ts +276 -308
  47. package/lib/annotations.ts +644 -644
  48. package/lib/build.ts +1766 -1766
  49. package/lib/citations.ts +160 -160
  50. package/lib/commands/build.ts +855 -855
  51. package/lib/commands/citations.ts +515 -515
  52. package/lib/commands/comments.ts +1050 -1050
  53. package/lib/commands/context.ts +176 -174
  54. package/lib/commands/core.ts +309 -309
  55. package/lib/commands/doi.ts +435 -435
  56. package/lib/commands/file-ops.ts +372 -372
  57. package/lib/commands/history.ts +320 -320
  58. package/lib/commands/index.ts +87 -87
  59. package/lib/commands/init.ts +259 -259
  60. package/lib/commands/merge-resolve.ts +378 -378
  61. package/lib/commands/preview.ts +178 -178
  62. package/lib/commands/project-info.ts +244 -244
  63. package/lib/commands/quality.ts +517 -517
  64. package/lib/commands/response.ts +454 -454
  65. package/lib/commands/section-boundaries.ts +82 -82
  66. package/lib/commands/sections.ts +451 -451
  67. package/lib/commands/sync.ts +709 -706
  68. package/lib/commands/text-ops.ts +449 -449
  69. package/lib/commands/utilities.ts +448 -448
  70. package/lib/commands/verify-anchors.ts +272 -272
  71. package/lib/commands/word-tools.ts +340 -340
  72. package/lib/comment-realign.ts +517 -517
  73. package/lib/config.ts +84 -84
  74. package/lib/crossref.ts +781 -781
  75. package/lib/csl.ts +191 -191
  76. package/lib/dependencies.ts +98 -98
  77. package/lib/diff-engine.ts +465 -465
  78. package/lib/doi-cache.ts +115 -115
  79. package/lib/doi.ts +897 -897
  80. package/lib/equations.ts +506 -506
  81. package/lib/errors.ts +346 -346
  82. package/lib/format.ts +541 -541
  83. package/lib/git.ts +326 -326
  84. package/lib/grammar.ts +303 -303
  85. package/lib/image-registry.ts +180 -180
  86. package/lib/import.ts +911 -911
  87. package/lib/journals.ts +543 -543
  88. package/lib/macro-filter.lua +201 -201
  89. package/lib/macros.ts +273 -273
  90. package/lib/merge.ts +633 -633
  91. package/lib/orcid.ts +144 -144
  92. package/lib/pdf-comments.ts +263 -263
  93. package/lib/pdf-import.ts +524 -524
  94. package/lib/plugins.ts +362 -362
  95. package/lib/postprocess.ts +188 -188
  96. package/lib/pptx-color-filter.lua +37 -37
  97. package/lib/pptx-template.ts +469 -469
  98. package/lib/pptx-themes.ts +483 -483
  99. package/lib/protect-restore.ts +520 -520
  100. package/lib/rate-limiter.ts +94 -94
  101. package/lib/response.ts +197 -197
  102. package/lib/restore-references.ts +240 -240
  103. package/lib/review.ts +327 -327
  104. package/lib/schema.ts +488 -488
  105. package/lib/scientific-words.ts +73 -73
  106. package/lib/sections.ts +425 -335
  107. package/lib/slides.ts +756 -756
  108. package/lib/spelling.ts +334 -334
  109. package/lib/templates.ts +526 -526
  110. package/lib/themes.ts +742 -742
  111. package/lib/trackchanges.ts +247 -247
  112. package/lib/tui.ts +450 -450
  113. package/lib/types.ts +550 -550
  114. package/lib/undo.ts +250 -250
  115. package/lib/utils.ts +69 -69
  116. package/lib/variables.ts +179 -179
  117. package/lib/word-extraction.ts +806 -806
  118. package/lib/word.ts +643 -643
  119. package/lib/wordcomments.ts +840 -840
  120. package/mkdocs.yml +64 -64
  121. package/package.json +137 -137
  122. package/scripts/postbuild.js +47 -47
  123. package/skill/REFERENCE.md +539 -539
  124. package/skill/SKILL.md +295 -295
  125. package/tsconfig.json +26 -26
  126. package/types/index.d.ts +525 -525
package/lib/errors.ts CHANGED
@@ -1,346 +1,346 @@
1
- /**
2
- * Error handling utilities with actionable suggestions
3
- */
4
-
5
- import chalk from 'chalk';
6
- import * as path from 'path';
7
- import * as fs from 'fs';
8
-
9
- interface BuildContext {
10
- bibPath?: string;
11
- format?: string;
12
- }
13
-
14
- /**
15
- * Format an error message with optional suggestions
16
- * @param message - Main error message
17
- * @param suggestions - Actionable suggestions
18
- * @returns Formatted error string
19
- */
20
- export function formatError(message: string, suggestions: string[] = []): string {
21
- const lines = [chalk.red(`Error: ${message}`)];
22
-
23
- if (suggestions.length > 0) {
24
- lines.push('');
25
- for (const suggestion of suggestions) {
26
- lines.push(chalk.dim(` ${suggestion}`));
27
- }
28
- }
29
-
30
- return lines.join('\n');
31
- }
32
-
33
- /**
34
- * Get actionable suggestions for file not found errors
35
- * @param filePath - The file path that wasn't found
36
- * @returns Array of suggestions
37
- */
38
- export function getFileNotFoundSuggestions(filePath: string): string[] {
39
- const suggestions: string[] = [];
40
- const ext = path.extname(filePath).toLowerCase();
41
- const dir = path.dirname(filePath);
42
- const base = path.basename(filePath);
43
-
44
- // Check if directory exists
45
- if (!fs.existsSync(dir)) {
46
- suggestions.push(`Directory does not exist: ${dir}`);
47
- suggestions.push(`Create it with: mkdir -p "${dir}"`);
48
- return suggestions;
49
- }
50
-
51
- // Look for similar files
52
- try {
53
- const files = fs.readdirSync(dir);
54
- const similar = findSimilarFiles(base, files, 3);
55
-
56
- if (similar.length > 0) {
57
- suggestions.push('Did you mean:');
58
- for (const f of similar) {
59
- suggestions.push(` ${path.join(dir, f)}`);
60
- }
61
- }
62
- } catch {
63
- // Directory not readable
64
- }
65
-
66
- // Extension-specific suggestions
67
- if (ext === '.md' || ext === '') {
68
- suggestions.push('Run "rev status" to see files in the current project');
69
- } else if (ext === '.docx') {
70
- suggestions.push('Use "rev import <docx>" to import Word documents');
71
- } else if (ext === '.bib') {
72
- suggestions.push('Create a bibliography with "rev doi bib <doi>"');
73
- suggestions.push('Or check references.bib in your project');
74
- } else if (ext === '.pdf') {
75
- suggestions.push('Build PDFs with "rev build pdf"');
76
- }
77
-
78
- return suggestions;
79
- }
80
-
81
- /**
82
- * Get actionable suggestions for dependency errors
83
- * @param dependency - The missing dependency
84
- * @returns Array of suggestions
85
- */
86
- export function getDependencySuggestions(dependency: string): string[] {
87
- const suggestions: string[] = [];
88
- const platform = process.platform;
89
-
90
- switch (dependency.toLowerCase()) {
91
- case 'pandoc':
92
- suggestions.push('Pandoc is required for document conversion');
93
- if (platform === 'darwin') {
94
- suggestions.push('Install with: brew install pandoc');
95
- } else if (platform === 'win32') {
96
- suggestions.push('Install from: https://pandoc.org/installing.html');
97
- suggestions.push('Or with: winget install --id JohnMacFarlane.Pandoc');
98
- } else {
99
- suggestions.push('Install with: sudo apt install pandoc');
100
- suggestions.push('Or from: https://pandoc.org/installing.html');
101
- }
102
- suggestions.push('Run "rev install" to check all dependencies');
103
- break;
104
-
105
- case 'pdflatex':
106
- case 'xelatex':
107
- case 'latex':
108
- suggestions.push('LaTeX is required for PDF generation');
109
- if (platform === 'darwin') {
110
- suggestions.push('Install with: brew install --cask mactex');
111
- suggestions.push('Or minimal: brew install --cask basictex');
112
- } else if (platform === 'win32') {
113
- suggestions.push('Install MiKTeX from: https://miktex.org/download');
114
- suggestions.push('Or TeX Live from: https://tug.org/texlive/');
115
- } else {
116
- suggestions.push('Install with: sudo apt install texlive-full');
117
- suggestions.push('Or minimal: sudo apt install texlive-latex-base');
118
- }
119
- suggestions.push('Alternative: Use "rev build docx" for Word output');
120
- break;
121
-
122
- case 'pandoc-crossref':
123
- suggestions.push('pandoc-crossref enables figure/table/equation numbering');
124
- if (platform === 'darwin') {
125
- suggestions.push('Install with: brew install pandoc-crossref');
126
- } else if (platform === 'win32') {
127
- suggestions.push('Download from: https://github.com/lierdakil/pandoc-crossref/releases');
128
- } else {
129
- suggestions.push('Install with: sudo apt install pandoc-crossref');
130
- suggestions.push('Or from: https://github.com/lierdakil/pandoc-crossref/releases');
131
- }
132
- suggestions.push('Cross-references will work but wonʼt be numbered without it');
133
- break;
134
- }
135
-
136
- return suggestions;
137
- }
138
-
139
- /**
140
- * Get actionable suggestions for configuration errors
141
- * @param field - The problematic config field
142
- * @param issue - What's wrong with it
143
- * @returns Array of suggestions
144
- */
145
- export function getConfigSuggestions(field: string, issue: string): string[] {
146
- const suggestions: string[] = [];
147
-
148
- switch (field) {
149
- case 'bibliography':
150
- suggestions.push('Create a references.bib file in your project');
151
- suggestions.push('Or set bibliography in rev.yaml:');
152
- suggestions.push(' bibliography: path/to/refs.bib');
153
- break;
154
-
155
- case 'sections':
156
- suggestions.push('List your sections in rev.yaml:');
157
- suggestions.push(' sections:');
158
- suggestions.push(' - introduction.md');
159
- suggestions.push(' - methods.md');
160
- suggestions.push('Or run "rev init" to auto-detect');
161
- break;
162
-
163
- case 'user':
164
- suggestions.push('Set your name for comment attribution:');
165
- suggestions.push(' rev config user "Your Name"');
166
- break;
167
-
168
- case 'csl':
169
- suggestions.push('CSL styles control citation format');
170
- suggestions.push('Download styles from: https://www.zotero.org/styles');
171
- suggestions.push('Or use: citation-style: apa (common styles available)');
172
- break;
173
-
174
- default:
175
- suggestions.push(`Check rev.yaml for "${field}" configuration`);
176
- suggestions.push('Run "rev help config" for configuration options');
177
- }
178
-
179
- if (issue === 'typo') {
180
- suggestions.unshift('This looks like a typo in rev.yaml');
181
- }
182
-
183
- return suggestions;
184
- }
185
-
186
- /**
187
- * Get suggestions for comment/annotation errors
188
- * @param issue - The issue type
189
- * @returns Array of suggestions
190
- */
191
- export function getAnnotationSuggestions(issue: string): string[] {
192
- const suggestions: string[] = [];
193
-
194
- switch (issue) {
195
- case 'no_comments':
196
- suggestions.push('Comments use CriticMarkup syntax:');
197
- suggestions.push(' {>>Author: Comment text<<}');
198
- suggestions.push('Import from Word with: rev import <docx>');
199
- break;
200
-
201
- case 'no_changes':
202
- suggestions.push('Track changes use CriticMarkup syntax:');
203
- suggestions.push(' {++inserted text++}');
204
- suggestions.push(' {--deleted text--}');
205
- suggestions.push(' {~~old~>new~~}');
206
- suggestions.push('Import from Word with: rev import <docx>');
207
- break;
208
-
209
- case 'invalid_number':
210
- suggestions.push('Use "rev comments <file>" to see comment numbers');
211
- suggestions.push('Or "rev status <file>" for a summary');
212
- break;
213
-
214
- case 'no_author':
215
- suggestions.push('Set your author name:');
216
- suggestions.push(' rev config user "Your Name"');
217
- suggestions.push('Or use --author "Name" flag');
218
- break;
219
- }
220
-
221
- return suggestions;
222
- }
223
-
224
- /**
225
- * Get suggestions for build errors
226
- * @param issue - The build issue
227
- * @param context - Additional context
228
- * @returns Array of suggestions
229
- */
230
- export function getBuildSuggestions(issue: string, context: BuildContext = {}): string[] {
231
- const suggestions: string[] = [];
232
-
233
- switch (issue) {
234
- case 'no_sections':
235
- suggestions.push('No section files found to build');
236
- suggestions.push('Create markdown files or run "rev new" to start a project');
237
- suggestions.push('Or run "rev init" to auto-detect existing files');
238
- break;
239
-
240
- case 'missing_bib':
241
- suggestions.push('Bibliography file not found');
242
- if (context.bibPath) {
243
- suggestions.push(`Expected: ${context.bibPath}`);
244
- }
245
- suggestions.push('Create references.bib or update rev.yaml');
246
- suggestions.push('Add citations with: rev doi bib <doi>');
247
- break;
248
-
249
- case 'pandoc_failed':
250
- suggestions.push('Pandoc conversion failed');
251
- suggestions.push('Check for syntax errors in your markdown');
252
- suggestions.push('Run "rev validate" to check document structure');
253
- if (context.format === 'pdf') {
254
- suggestions.push('Try "rev build docx" as an alternative');
255
- }
256
- break;
257
-
258
- case 'latex_error':
259
- suggestions.push('LaTeX compilation failed');
260
- suggestions.push('Common issues:');
261
- suggestions.push(' - Missing packages (run tlmgr to install)');
262
- suggestions.push(' - Invalid characters in text');
263
- suggestions.push(' - Math mode errors');
264
- suggestions.push('Try "rev build docx" to bypass LaTeX');
265
- break;
266
- }
267
-
268
- return suggestions;
269
- }
270
-
271
- /**
272
- * Find similar filenames using Levenshtein distance
273
- * @param target - Target filename
274
- * @param candidates - Available filenames
275
- * @param limit - Max results
276
- * @returns Array of similar filenames
277
- */
278
- function findSimilarFiles(target: string, candidates: string[], limit: number = 3): string[] {
279
- const scored = candidates
280
- .map(c => ({ name: c, distance: levenshtein(target.toLowerCase(), c.toLowerCase()) }))
281
- .filter(c => c.distance <= 3) // Only reasonably similar
282
- .sort((a, b) => a.distance - b.distance);
283
-
284
- return scored.slice(0, limit).map(c => c.name);
285
- }
286
-
287
- /**
288
- * Simple Levenshtein distance
289
- * @param a - First string
290
- * @param b - Second string
291
- * @returns Edit distance
292
- */
293
- function levenshtein(a: string, b: string): number {
294
- if (a.length === 0) return b.length;
295
- if (b.length === 0) return a.length;
296
-
297
- const matrix: number[][] = [];
298
-
299
- for (let i = 0; i <= b.length; i++) {
300
- matrix[i] = [i];
301
- }
302
-
303
- for (let j = 0; j <= a.length; j++) {
304
- matrix[0]![j] = j;
305
- }
306
-
307
- for (let i = 1; i <= b.length; i++) {
308
- for (let j = 1; j <= a.length; j++) {
309
- if (b.charAt(i - 1) === a.charAt(j - 1)) {
310
- matrix[i]![j] = matrix[i - 1]![j - 1]!;
311
- } else {
312
- matrix[i]![j] = Math.min(
313
- matrix[i - 1]![j - 1]! + 1,
314
- matrix[i]![j - 1]! + 1,
315
- matrix[i - 1]![j]! + 1
316
- );
317
- }
318
- }
319
- }
320
-
321
- return matrix[b.length]![a.length]!;
322
- }
323
-
324
- /**
325
- * Print error and exit
326
- * @param message - Error message
327
- * @param suggestions - Suggestions
328
- */
329
- export function exitWithError(message: string, suggestions: string[] = []): never {
330
- console.error(formatError(message, suggestions));
331
- process.exit(1);
332
- }
333
-
334
- /**
335
- * Validate file exists with helpful error
336
- * @param filePath - File to check
337
- * @param fileType - Type description for error message
338
- */
339
- export function requireFile(filePath: string, fileType: string = 'File'): void {
340
- if (!fs.existsSync(filePath)) {
341
- exitWithError(
342
- `${fileType} not found: ${filePath}`,
343
- getFileNotFoundSuggestions(filePath)
344
- );
345
- }
346
- }
1
+ /**
2
+ * Error handling utilities with actionable suggestions
3
+ */
4
+
5
+ import chalk from 'chalk';
6
+ import * as path from 'path';
7
+ import * as fs from 'fs';
8
+
9
+ interface BuildContext {
10
+ bibPath?: string;
11
+ format?: string;
12
+ }
13
+
14
+ /**
15
+ * Format an error message with optional suggestions
16
+ * @param message - Main error message
17
+ * @param suggestions - Actionable suggestions
18
+ * @returns Formatted error string
19
+ */
20
+ export function formatError(message: string, suggestions: string[] = []): string {
21
+ const lines = [chalk.red(`Error: ${message}`)];
22
+
23
+ if (suggestions.length > 0) {
24
+ lines.push('');
25
+ for (const suggestion of suggestions) {
26
+ lines.push(chalk.dim(` ${suggestion}`));
27
+ }
28
+ }
29
+
30
+ return lines.join('\n');
31
+ }
32
+
33
+ /**
34
+ * Get actionable suggestions for file not found errors
35
+ * @param filePath - The file path that wasn't found
36
+ * @returns Array of suggestions
37
+ */
38
+ export function getFileNotFoundSuggestions(filePath: string): string[] {
39
+ const suggestions: string[] = [];
40
+ const ext = path.extname(filePath).toLowerCase();
41
+ const dir = path.dirname(filePath);
42
+ const base = path.basename(filePath);
43
+
44
+ // Check if directory exists
45
+ if (!fs.existsSync(dir)) {
46
+ suggestions.push(`Directory does not exist: ${dir}`);
47
+ suggestions.push(`Create it with: mkdir -p "${dir}"`);
48
+ return suggestions;
49
+ }
50
+
51
+ // Look for similar files
52
+ try {
53
+ const files = fs.readdirSync(dir);
54
+ const similar = findSimilarFiles(base, files, 3);
55
+
56
+ if (similar.length > 0) {
57
+ suggestions.push('Did you mean:');
58
+ for (const f of similar) {
59
+ suggestions.push(` ${path.join(dir, f)}`);
60
+ }
61
+ }
62
+ } catch {
63
+ // Directory not readable
64
+ }
65
+
66
+ // Extension-specific suggestions
67
+ if (ext === '.md' || ext === '') {
68
+ suggestions.push('Run "rev status" to see files in the current project');
69
+ } else if (ext === '.docx') {
70
+ suggestions.push('Use "rev import <docx>" to import Word documents');
71
+ } else if (ext === '.bib') {
72
+ suggestions.push('Create a bibliography with "rev doi bib <doi>"');
73
+ suggestions.push('Or check references.bib in your project');
74
+ } else if (ext === '.pdf') {
75
+ suggestions.push('Build PDFs with "rev build pdf"');
76
+ }
77
+
78
+ return suggestions;
79
+ }
80
+
81
+ /**
82
+ * Get actionable suggestions for dependency errors
83
+ * @param dependency - The missing dependency
84
+ * @returns Array of suggestions
85
+ */
86
+ export function getDependencySuggestions(dependency: string): string[] {
87
+ const suggestions: string[] = [];
88
+ const platform = process.platform;
89
+
90
+ switch (dependency.toLowerCase()) {
91
+ case 'pandoc':
92
+ suggestions.push('Pandoc is required for document conversion');
93
+ if (platform === 'darwin') {
94
+ suggestions.push('Install with: brew install pandoc');
95
+ } else if (platform === 'win32') {
96
+ suggestions.push('Install from: https://pandoc.org/installing.html');
97
+ suggestions.push('Or with: winget install --id JohnMacFarlane.Pandoc');
98
+ } else {
99
+ suggestions.push('Install with: sudo apt install pandoc');
100
+ suggestions.push('Or from: https://pandoc.org/installing.html');
101
+ }
102
+ suggestions.push('Run "rev install" to check all dependencies');
103
+ break;
104
+
105
+ case 'pdflatex':
106
+ case 'xelatex':
107
+ case 'latex':
108
+ suggestions.push('LaTeX is required for PDF generation');
109
+ if (platform === 'darwin') {
110
+ suggestions.push('Install with: brew install --cask mactex');
111
+ suggestions.push('Or minimal: brew install --cask basictex');
112
+ } else if (platform === 'win32') {
113
+ suggestions.push('Install MiKTeX from: https://miktex.org/download');
114
+ suggestions.push('Or TeX Live from: https://tug.org/texlive/');
115
+ } else {
116
+ suggestions.push('Install with: sudo apt install texlive-full');
117
+ suggestions.push('Or minimal: sudo apt install texlive-latex-base');
118
+ }
119
+ suggestions.push('Alternative: Use "rev build docx" for Word output');
120
+ break;
121
+
122
+ case 'pandoc-crossref':
123
+ suggestions.push('pandoc-crossref enables figure/table/equation numbering');
124
+ if (platform === 'darwin') {
125
+ suggestions.push('Install with: brew install pandoc-crossref');
126
+ } else if (platform === 'win32') {
127
+ suggestions.push('Download from: https://github.com/lierdakil/pandoc-crossref/releases');
128
+ } else {
129
+ suggestions.push('Install with: sudo apt install pandoc-crossref');
130
+ suggestions.push('Or from: https://github.com/lierdakil/pandoc-crossref/releases');
131
+ }
132
+ suggestions.push('Cross-references will work but wonʼt be numbered without it');
133
+ break;
134
+ }
135
+
136
+ return suggestions;
137
+ }
138
+
139
+ /**
140
+ * Get actionable suggestions for configuration errors
141
+ * @param field - The problematic config field
142
+ * @param issue - What's wrong with it
143
+ * @returns Array of suggestions
144
+ */
145
+ export function getConfigSuggestions(field: string, issue: string): string[] {
146
+ const suggestions: string[] = [];
147
+
148
+ switch (field) {
149
+ case 'bibliography':
150
+ suggestions.push('Create a references.bib file in your project');
151
+ suggestions.push('Or set bibliography in rev.yaml:');
152
+ suggestions.push(' bibliography: path/to/refs.bib');
153
+ break;
154
+
155
+ case 'sections':
156
+ suggestions.push('List your sections in rev.yaml:');
157
+ suggestions.push(' sections:');
158
+ suggestions.push(' - introduction.md');
159
+ suggestions.push(' - methods.md');
160
+ suggestions.push('Or run "rev init" to auto-detect');
161
+ break;
162
+
163
+ case 'user':
164
+ suggestions.push('Set your name for comment attribution:');
165
+ suggestions.push(' rev config user "Your Name"');
166
+ break;
167
+
168
+ case 'csl':
169
+ suggestions.push('CSL styles control citation format');
170
+ suggestions.push('Download styles from: https://www.zotero.org/styles');
171
+ suggestions.push('Or use: citation-style: apa (common styles available)');
172
+ break;
173
+
174
+ default:
175
+ suggestions.push(`Check rev.yaml for "${field}" configuration`);
176
+ suggestions.push('Run "rev help config" for configuration options');
177
+ }
178
+
179
+ if (issue === 'typo') {
180
+ suggestions.unshift('This looks like a typo in rev.yaml');
181
+ }
182
+
183
+ return suggestions;
184
+ }
185
+
186
+ /**
187
+ * Get suggestions for comment/annotation errors
188
+ * @param issue - The issue type
189
+ * @returns Array of suggestions
190
+ */
191
+ export function getAnnotationSuggestions(issue: string): string[] {
192
+ const suggestions: string[] = [];
193
+
194
+ switch (issue) {
195
+ case 'no_comments':
196
+ suggestions.push('Comments use CriticMarkup syntax:');
197
+ suggestions.push(' {>>Author: Comment text<<}');
198
+ suggestions.push('Import from Word with: rev import <docx>');
199
+ break;
200
+
201
+ case 'no_changes':
202
+ suggestions.push('Track changes use CriticMarkup syntax:');
203
+ suggestions.push(' {++inserted text++}');
204
+ suggestions.push(' {--deleted text--}');
205
+ suggestions.push(' {~~old~>new~~}');
206
+ suggestions.push('Import from Word with: rev import <docx>');
207
+ break;
208
+
209
+ case 'invalid_number':
210
+ suggestions.push('Use "rev comments <file>" to see comment numbers');
211
+ suggestions.push('Or "rev status <file>" for a summary');
212
+ break;
213
+
214
+ case 'no_author':
215
+ suggestions.push('Set your author name:');
216
+ suggestions.push(' rev config user "Your Name"');
217
+ suggestions.push('Or use --author "Name" flag');
218
+ break;
219
+ }
220
+
221
+ return suggestions;
222
+ }
223
+
224
+ /**
225
+ * Get suggestions for build errors
226
+ * @param issue - The build issue
227
+ * @param context - Additional context
228
+ * @returns Array of suggestions
229
+ */
230
+ export function getBuildSuggestions(issue: string, context: BuildContext = {}): string[] {
231
+ const suggestions: string[] = [];
232
+
233
+ switch (issue) {
234
+ case 'no_sections':
235
+ suggestions.push('No section files found to build');
236
+ suggestions.push('Create markdown files or run "rev new" to start a project');
237
+ suggestions.push('Or run "rev init" to auto-detect existing files');
238
+ break;
239
+
240
+ case 'missing_bib':
241
+ suggestions.push('Bibliography file not found');
242
+ if (context.bibPath) {
243
+ suggestions.push(`Expected: ${context.bibPath}`);
244
+ }
245
+ suggestions.push('Create references.bib or update rev.yaml');
246
+ suggestions.push('Add citations with: rev doi bib <doi>');
247
+ break;
248
+
249
+ case 'pandoc_failed':
250
+ suggestions.push('Pandoc conversion failed');
251
+ suggestions.push('Check for syntax errors in your markdown');
252
+ suggestions.push('Run "rev validate" to check document structure');
253
+ if (context.format === 'pdf') {
254
+ suggestions.push('Try "rev build docx" as an alternative');
255
+ }
256
+ break;
257
+
258
+ case 'latex_error':
259
+ suggestions.push('LaTeX compilation failed');
260
+ suggestions.push('Common issues:');
261
+ suggestions.push(' - Missing packages (run tlmgr to install)');
262
+ suggestions.push(' - Invalid characters in text');
263
+ suggestions.push(' - Math mode errors');
264
+ suggestions.push('Try "rev build docx" to bypass LaTeX');
265
+ break;
266
+ }
267
+
268
+ return suggestions;
269
+ }
270
+
271
+ /**
272
+ * Find similar filenames using Levenshtein distance
273
+ * @param target - Target filename
274
+ * @param candidates - Available filenames
275
+ * @param limit - Max results
276
+ * @returns Array of similar filenames
277
+ */
278
+ function findSimilarFiles(target: string, candidates: string[], limit: number = 3): string[] {
279
+ const scored = candidates
280
+ .map(c => ({ name: c, distance: levenshtein(target.toLowerCase(), c.toLowerCase()) }))
281
+ .filter(c => c.distance <= 3) // Only reasonably similar
282
+ .sort((a, b) => a.distance - b.distance);
283
+
284
+ return scored.slice(0, limit).map(c => c.name);
285
+ }
286
+
287
+ /**
288
+ * Simple Levenshtein distance
289
+ * @param a - First string
290
+ * @param b - Second string
291
+ * @returns Edit distance
292
+ */
293
+ function levenshtein(a: string, b: string): number {
294
+ if (a.length === 0) return b.length;
295
+ if (b.length === 0) return a.length;
296
+
297
+ const matrix: number[][] = [];
298
+
299
+ for (let i = 0; i <= b.length; i++) {
300
+ matrix[i] = [i];
301
+ }
302
+
303
+ for (let j = 0; j <= a.length; j++) {
304
+ matrix[0]![j] = j;
305
+ }
306
+
307
+ for (let i = 1; i <= b.length; i++) {
308
+ for (let j = 1; j <= a.length; j++) {
309
+ if (b.charAt(i - 1) === a.charAt(j - 1)) {
310
+ matrix[i]![j] = matrix[i - 1]![j - 1]!;
311
+ } else {
312
+ matrix[i]![j] = Math.min(
313
+ matrix[i - 1]![j - 1]! + 1,
314
+ matrix[i]![j - 1]! + 1,
315
+ matrix[i - 1]![j]! + 1
316
+ );
317
+ }
318
+ }
319
+ }
320
+
321
+ return matrix[b.length]![a.length]!;
322
+ }
323
+
324
+ /**
325
+ * Print error and exit
326
+ * @param message - Error message
327
+ * @param suggestions - Suggestions
328
+ */
329
+ export function exitWithError(message: string, suggestions: string[] = []): never {
330
+ console.error(formatError(message, suggestions));
331
+ process.exit(1);
332
+ }
333
+
334
+ /**
335
+ * Validate file exists with helpful error
336
+ * @param filePath - File to check
337
+ * @param fileType - Type description for error message
338
+ */
339
+ export function requireFile(filePath: string, fileType: string = 'File'): void {
340
+ if (!fs.existsSync(filePath)) {
341
+ exitWithError(
342
+ `${fileType} not found: ${filePath}`,
343
+ getFileNotFoundSuggestions(filePath)
344
+ );
345
+ }
346
+ }