@wundr.io/cli 1.0.0 → 1.0.2-dev.20260530174250.ef0ec927

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 (230) hide show
  1. package/README.md +696 -280
  2. package/bin/wundr.js +13 -5
  3. package/package.json +30 -9
  4. package/src/ai/ai-service.ts +6 -4
  5. package/src/ai/claude-client.ts +6 -2
  6. package/src/ai/conversation-manager.ts +12 -5
  7. package/src/cli.ts +42 -13
  8. package/src/commands/ai.ts +340 -64
  9. package/src/commands/alignment.ts +1212 -0
  10. package/src/commands/analyze-optimized.ts +371 -33
  11. package/src/commands/analyze.ts +8 -6
  12. package/src/commands/batch.ts +166 -26
  13. package/src/commands/chat.ts +20 -10
  14. package/src/commands/claude-init.ts +31 -27
  15. package/src/commands/claude-setup.ts +761 -81
  16. package/src/commands/computer-setup.ts +524 -12
  17. package/src/commands/create-command.ts +3 -3
  18. package/src/commands/create.ts +9 -6
  19. package/src/commands/dashboard.ts +11 -6
  20. package/src/commands/govern.ts +11 -6
  21. package/src/commands/governance.ts +1005 -0
  22. package/src/commands/guardian.ts +887 -0
  23. package/src/commands/init.ts +104 -11
  24. package/src/commands/orchestrator.ts +789 -0
  25. package/src/commands/performance-optimizer.ts +15 -10
  26. package/src/commands/plugins.ts +8 -5
  27. package/src/commands/project-update.ts +1156 -0
  28. package/src/commands/rag.ts +1011 -0
  29. package/src/commands/session.ts +631 -0
  30. package/src/commands/setup.ts +42 -344
  31. package/src/commands/test-init.ts +3 -2
  32. package/src/commands/test.ts +3 -2
  33. package/src/commands/watch.ts +21 -11
  34. package/src/commands/worktree.ts +1057 -0
  35. package/src/context/context-manager.ts +5 -2
  36. package/src/context/session-manager.ts +18 -7
  37. package/src/framework/command-interface.ts +520 -0
  38. package/src/framework/command-registry.ts +942 -0
  39. package/src/framework/completion-exporter.ts +383 -0
  40. package/src/framework/debug-logger.ts +519 -0
  41. package/src/framework/error-handler.ts +867 -0
  42. package/src/framework/help-generator.ts +540 -0
  43. package/src/framework/index.ts +169 -0
  44. package/src/framework/interactive-repl.ts +703 -0
  45. package/src/framework/output-formatter.ts +834 -0
  46. package/src/framework/progress-manager.ts +539 -0
  47. package/src/index.ts +3 -2
  48. package/src/interactive/interactive-mode.ts +14 -7
  49. package/src/lib/conflict-resolution.ts +818 -0
  50. package/src/lib/merge-strategy.ts +550 -0
  51. package/src/lib/safety-mechanisms.ts +451 -0
  52. package/src/lib/state-detection.ts +1030 -0
  53. package/src/nlp/command-mapper.ts +8 -3
  54. package/src/nlp/command-parser.ts +5 -2
  55. package/src/nlp/intent-parser.ts +23 -9
  56. package/src/plugins/plugin-manager.ts +50 -24
  57. package/src/tests/computer-setup-integration.test.ts +470 -0
  58. package/src/types/index.ts +1 -1
  59. package/src/types/modules.d.ts +425 -1
  60. package/src/utils/backup-rollback-manager.ts +366 -0
  61. package/src/utils/claude-config-installer.ts +823 -0
  62. package/src/utils/config-manager.ts +9 -6
  63. package/src/utils/error-handler.ts +3 -1
  64. package/src/utils/logger.ts +35 -12
  65. package/templates/batch/ci-cd.yaml +7 -7
  66. package/test-suites/api/health.spec.ts +20 -23
  67. package/test-suites/helpers/test-config.ts +14 -13
  68. package/test-suites/ui/accessibility.spec.ts +27 -22
  69. package/test-suites/ui/smoke.spec.ts +26 -21
  70. package/dist/ai/ai-service.d.ts +0 -152
  71. package/dist/ai/ai-service.d.ts.map +0 -1
  72. package/dist/ai/ai-service.js +0 -430
  73. package/dist/ai/ai-service.js.map +0 -1
  74. package/dist/ai/claude-client.d.ts +0 -130
  75. package/dist/ai/claude-client.d.ts.map +0 -1
  76. package/dist/ai/claude-client.js +0 -339
  77. package/dist/ai/claude-client.js.map +0 -1
  78. package/dist/ai/conversation-manager.d.ts +0 -164
  79. package/dist/ai/conversation-manager.d.ts.map +0 -1
  80. package/dist/ai/conversation-manager.js +0 -612
  81. package/dist/ai/conversation-manager.js.map +0 -1
  82. package/dist/ai/index.d.ts +0 -5
  83. package/dist/ai/index.d.ts.map +0 -1
  84. package/dist/ai/index.js +0 -8
  85. package/dist/ai/index.js.map +0 -1
  86. package/dist/cli.d.ts +0 -36
  87. package/dist/cli.d.ts.map +0 -1
  88. package/dist/cli.js +0 -173
  89. package/dist/cli.js.map +0 -1
  90. package/dist/commands/ai.d.ts +0 -89
  91. package/dist/commands/ai.d.ts.map +0 -1
  92. package/dist/commands/ai.js +0 -735
  93. package/dist/commands/ai.js.map +0 -1
  94. package/dist/commands/analyze-optimized.d.ts +0 -14
  95. package/dist/commands/analyze-optimized.d.ts.map +0 -1
  96. package/dist/commands/analyze-optimized.js +0 -437
  97. package/dist/commands/analyze-optimized.js.map +0 -1
  98. package/dist/commands/analyze.d.ts +0 -65
  99. package/dist/commands/analyze.d.ts.map +0 -1
  100. package/dist/commands/analyze.js +0 -435
  101. package/dist/commands/analyze.js.map +0 -1
  102. package/dist/commands/batch.d.ts +0 -71
  103. package/dist/commands/batch.d.ts.map +0 -1
  104. package/dist/commands/batch.js +0 -738
  105. package/dist/commands/batch.js.map +0 -1
  106. package/dist/commands/chat.d.ts +0 -71
  107. package/dist/commands/chat.d.ts.map +0 -1
  108. package/dist/commands/chat.js +0 -674
  109. package/dist/commands/chat.js.map +0 -1
  110. package/dist/commands/claude-init.d.ts +0 -28
  111. package/dist/commands/claude-init.d.ts.map +0 -1
  112. package/dist/commands/claude-init.js +0 -587
  113. package/dist/commands/claude-init.js.map +0 -1
  114. package/dist/commands/claude-setup.d.ts +0 -32
  115. package/dist/commands/claude-setup.d.ts.map +0 -1
  116. package/dist/commands/claude-setup.js +0 -570
  117. package/dist/commands/claude-setup.js.map +0 -1
  118. package/dist/commands/computer-setup-commands.d.ts +0 -39
  119. package/dist/commands/computer-setup-commands.d.ts.map +0 -1
  120. package/dist/commands/computer-setup-commands.js +0 -563
  121. package/dist/commands/computer-setup-commands.js.map +0 -1
  122. package/dist/commands/computer-setup.d.ts +0 -7
  123. package/dist/commands/computer-setup.d.ts.map +0 -1
  124. package/dist/commands/computer-setup.js +0 -481
  125. package/dist/commands/computer-setup.js.map +0 -1
  126. package/dist/commands/create-command.d.ts +0 -7
  127. package/dist/commands/create-command.d.ts.map +0 -1
  128. package/dist/commands/create-command.js +0 -158
  129. package/dist/commands/create-command.js.map +0 -1
  130. package/dist/commands/create.d.ts +0 -74
  131. package/dist/commands/create.d.ts.map +0 -1
  132. package/dist/commands/create.js +0 -556
  133. package/dist/commands/create.js.map +0 -1
  134. package/dist/commands/dashboard.d.ts +0 -91
  135. package/dist/commands/dashboard.d.ts.map +0 -1
  136. package/dist/commands/dashboard.js +0 -537
  137. package/dist/commands/dashboard.js.map +0 -1
  138. package/dist/commands/govern.d.ts +0 -70
  139. package/dist/commands/govern.d.ts.map +0 -1
  140. package/dist/commands/govern.js +0 -480
  141. package/dist/commands/govern.js.map +0 -1
  142. package/dist/commands/init.d.ts +0 -55
  143. package/dist/commands/init.d.ts.map +0 -1
  144. package/dist/commands/init.js +0 -584
  145. package/dist/commands/init.js.map +0 -1
  146. package/dist/commands/performance-optimizer.d.ts +0 -30
  147. package/dist/commands/performance-optimizer.d.ts.map +0 -1
  148. package/dist/commands/performance-optimizer.js +0 -649
  149. package/dist/commands/performance-optimizer.js.map +0 -1
  150. package/dist/commands/plugins.d.ts +0 -87
  151. package/dist/commands/plugins.d.ts.map +0 -1
  152. package/dist/commands/plugins.js +0 -685
  153. package/dist/commands/plugins.js.map +0 -1
  154. package/dist/commands/setup.d.ts +0 -29
  155. package/dist/commands/setup.d.ts.map +0 -1
  156. package/dist/commands/setup.js +0 -399
  157. package/dist/commands/setup.js.map +0 -1
  158. package/dist/commands/test-init.d.ts +0 -9
  159. package/dist/commands/test-init.d.ts.map +0 -1
  160. package/dist/commands/test-init.js +0 -222
  161. package/dist/commands/test-init.js.map +0 -1
  162. package/dist/commands/test.d.ts +0 -25
  163. package/dist/commands/test.d.ts.map +0 -1
  164. package/dist/commands/test.js +0 -217
  165. package/dist/commands/test.js.map +0 -1
  166. package/dist/commands/watch.d.ts +0 -76
  167. package/dist/commands/watch.d.ts.map +0 -1
  168. package/dist/commands/watch.js +0 -610
  169. package/dist/commands/watch.js.map +0 -1
  170. package/dist/context/context-manager.d.ts +0 -155
  171. package/dist/context/context-manager.d.ts.map +0 -1
  172. package/dist/context/context-manager.js +0 -383
  173. package/dist/context/context-manager.js.map +0 -1
  174. package/dist/context/index.d.ts +0 -3
  175. package/dist/context/index.d.ts.map +0 -1
  176. package/dist/context/index.js +0 -6
  177. package/dist/context/index.js.map +0 -1
  178. package/dist/context/session-manager.d.ts +0 -207
  179. package/dist/context/session-manager.d.ts.map +0 -1
  180. package/dist/context/session-manager.js +0 -682
  181. package/dist/context/session-manager.js.map +0 -1
  182. package/dist/index.d.ts +0 -8
  183. package/dist/index.d.ts.map +0 -1
  184. package/dist/index.js +0 -51
  185. package/dist/index.js.map +0 -1
  186. package/dist/interactive/interactive-mode.d.ts +0 -76
  187. package/dist/interactive/interactive-mode.d.ts.map +0 -1
  188. package/dist/interactive/interactive-mode.js +0 -730
  189. package/dist/interactive/interactive-mode.js.map +0 -1
  190. package/dist/nlp/command-mapper.d.ts +0 -174
  191. package/dist/nlp/command-mapper.d.ts.map +0 -1
  192. package/dist/nlp/command-mapper.js +0 -623
  193. package/dist/nlp/command-mapper.js.map +0 -1
  194. package/dist/nlp/command-parser.d.ts +0 -106
  195. package/dist/nlp/command-parser.d.ts.map +0 -1
  196. package/dist/nlp/command-parser.js +0 -416
  197. package/dist/nlp/command-parser.js.map +0 -1
  198. package/dist/nlp/index.d.ts +0 -5
  199. package/dist/nlp/index.d.ts.map +0 -1
  200. package/dist/nlp/index.js +0 -8
  201. package/dist/nlp/index.js.map +0 -1
  202. package/dist/nlp/intent-classifier.d.ts +0 -59
  203. package/dist/nlp/intent-classifier.d.ts.map +0 -1
  204. package/dist/nlp/intent-classifier.js +0 -384
  205. package/dist/nlp/intent-classifier.js.map +0 -1
  206. package/dist/nlp/intent-parser.d.ts +0 -152
  207. package/dist/nlp/intent-parser.d.ts.map +0 -1
  208. package/dist/nlp/intent-parser.js +0 -739
  209. package/dist/nlp/intent-parser.js.map +0 -1
  210. package/dist/plugins/plugin-manager.d.ts +0 -120
  211. package/dist/plugins/plugin-manager.d.ts.map +0 -1
  212. package/dist/plugins/plugin-manager.js +0 -595
  213. package/dist/plugins/plugin-manager.js.map +0 -1
  214. package/dist/types/index.d.ts +0 -224
  215. package/dist/types/index.d.ts.map +0 -1
  216. package/dist/types/index.js +0 -3
  217. package/dist/types/index.js.map +0 -1
  218. package/dist/utils/config-manager.d.ts +0 -73
  219. package/dist/utils/config-manager.d.ts.map +0 -1
  220. package/dist/utils/config-manager.js +0 -339
  221. package/dist/utils/config-manager.js.map +0 -1
  222. package/dist/utils/error-handler.d.ts +0 -46
  223. package/dist/utils/error-handler.d.ts.map +0 -1
  224. package/dist/utils/error-handler.js +0 -169
  225. package/dist/utils/error-handler.js.map +0 -1
  226. package/dist/utils/logger.d.ts +0 -25
  227. package/dist/utils/logger.d.ts.map +0 -1
  228. package/dist/utils/logger.js +0 -94
  229. package/dist/utils/logger.js.map +0 -1
  230. package/src/commands/computer-setup-commands.ts +0 -709
@@ -0,0 +1,451 @@
1
+ /**
2
+ * Safety Mechanisms - File-system based backup, transaction, and rollback support
3
+ *
4
+ * Backups are stored under `{projectRoot}/.wundr-backup/{backupId}/` and a
5
+ * metadata index is maintained at `{projectRoot}/.wundr-backup/index.json`.
6
+ *
7
+ * Transaction pattern:
8
+ * const tx = safetyManager.startTransaction('label');
9
+ * tx.recordOperation({ type, path, backupRef });
10
+ * tx.completeOperation(path);
11
+ * await tx.commit(); // no-op for now; marks the transaction done
12
+ * // or
13
+ * await tx.rollback(); // restores each file from its backupRef
14
+ */
15
+
16
+ import { existsSync } from 'fs';
17
+ import * as fs from 'fs/promises';
18
+ import * as path from 'path';
19
+
20
+ // ---------------------------------------------------------------------------
21
+ // Public interfaces (kept from original stub + extended for consumer usage)
22
+ // ---------------------------------------------------------------------------
23
+
24
+ export interface SafetyManagerOptions {
25
+ /** Root directory of the project being managed */
26
+ projectRoot?: string;
27
+ /** When true, skip writing any backup files to disk */
28
+ skipBackup?: boolean;
29
+ /** When true, do not write anything to disk */
30
+ dryRun?: boolean;
31
+ }
32
+
33
+ export interface SafetyManager {
34
+ /** Create a full backup of the supplied file list */
35
+ createBackup(
36
+ files: string[],
37
+ description?: string,
38
+ fromVersion?: string,
39
+ toVersion?: string
40
+ ): Promise<UpdateBackup>;
41
+ /** Begin a logical transaction that can be committed or rolled back */
42
+ startTransaction(label?: string): UpdateTransaction;
43
+ /** List all stored backups, newest first */
44
+ listBackups(): Promise<UpdateBackup[]>;
45
+ /** Return the most recent backup, or null if none exist */
46
+ getLatestBackup(): Promise<UpdateBackup | null>;
47
+ /**
48
+ * Restore files from a backup.
49
+ * Accepts either a full UpdateBackup object or a bare backup-id string
50
+ * so that both usages in project-update.ts are satisfied.
51
+ */
52
+ restoreFromBackup(backupOrId: UpdateBackup | string): Promise<boolean>;
53
+ /** Permanently remove a backup by id */
54
+ deleteBackup(backupId: string): Promise<boolean>;
55
+ // --- Legacy interface aliases kept for backwards-compatibility ---
56
+ /** @deprecated Use startTransaction instead */
57
+ beginTransaction(): UpdateTransaction;
58
+ }
59
+
60
+ export interface UpdateBackup {
61
+ id: string;
62
+ timestamp: Date;
63
+ /** Absolute path to the backup directory on disk */
64
+ path: string;
65
+ /** List of original file paths that were backed up */
66
+ files: string[];
67
+ description: string;
68
+ fromVersion: string;
69
+ toVersion: string;
70
+ /** Convenience method that delegates to SafetyManager.restoreFromBackup */
71
+ restore(): Promise<void>;
72
+ }
73
+
74
+ export interface TransactionOperation {
75
+ type: 'update' | 'create' | 'delete';
76
+ path: string;
77
+ /** Reference to the backup entry that holds the pre-change copy */
78
+ backupRef: string | null;
79
+ }
80
+
81
+ export interface UpdateTransaction {
82
+ /** Record a pending operation; call before modifying the file */
83
+ recordOperation(op: TransactionOperation): void;
84
+ /** Mark a previously-recorded operation as successfully completed */
85
+ completeOperation(filePath: string): void;
86
+ /** Mark a previously-recorded operation as failed */
87
+ failOperation(filePath: string, reason: string): void;
88
+ /** Commit the transaction (clears the in-memory log) */
89
+ commit(): Promise<boolean>;
90
+ /** Roll back by restoring each file from its saved backup copy */
91
+ rollback(): Promise<void>;
92
+ }
93
+
94
+ // ---------------------------------------------------------------------------
95
+ // Serialised form stored in index.json
96
+ // ---------------------------------------------------------------------------
97
+
98
+ interface BackupIndexEntry {
99
+ id: string;
100
+ timestamp: string; // ISO string
101
+ path: string;
102
+ files: string[];
103
+ description: string;
104
+ fromVersion: string;
105
+ toVersion: string;
106
+ }
107
+
108
+ // ---------------------------------------------------------------------------
109
+ // Internal helpers
110
+ // ---------------------------------------------------------------------------
111
+
112
+ function generateBackupId(): string {
113
+ // e.g. "bkp-20240311T143022-456"
114
+ const ts = new Date().toISOString().replace(/[:.]/g, '').slice(0, 15);
115
+ const rand = Math.floor(Math.random() * 1000)
116
+ .toString()
117
+ .padStart(3, '0');
118
+ return `bkp-${ts}-${rand}`;
119
+ }
120
+
121
+ function backupDir(projectRoot: string): string {
122
+ return path.join(projectRoot, '.wundr-backup');
123
+ }
124
+
125
+ function indexPath(projectRoot: string): string {
126
+ return path.join(backupDir(projectRoot), 'index.json');
127
+ }
128
+
129
+ async function ensureBackupDir(projectRoot: string): Promise<void> {
130
+ await fs.mkdir(backupDir(projectRoot), { recursive: true });
131
+
132
+ const idx = indexPath(projectRoot);
133
+ if (!existsSync(idx)) {
134
+ await fs.writeFile(idx, JSON.stringify([], null, 2), 'utf-8');
135
+ }
136
+ }
137
+
138
+ async function readIndex(projectRoot: string): Promise<BackupIndexEntry[]> {
139
+ try {
140
+ const raw = await fs.readFile(indexPath(projectRoot), 'utf-8');
141
+ return JSON.parse(raw) as BackupIndexEntry[];
142
+ } catch {
143
+ return [];
144
+ }
145
+ }
146
+
147
+ async function writeIndex(
148
+ projectRoot: string,
149
+ entries: BackupIndexEntry[]
150
+ ): Promise<void> {
151
+ await fs.writeFile(
152
+ indexPath(projectRoot),
153
+ JSON.stringify(entries, null, 2),
154
+ 'utf-8'
155
+ );
156
+ }
157
+
158
+ /**
159
+ * Convert an index entry + projectRoot into a full UpdateBackup object.
160
+ */
161
+ function entryToBackup(
162
+ entry: BackupIndexEntry,
163
+ manager: SafetyManager
164
+ ): UpdateBackup {
165
+ return {
166
+ id: entry.id,
167
+ timestamp: new Date(entry.timestamp),
168
+ path: entry.path,
169
+ files: entry.files,
170
+ description: entry.description,
171
+ fromVersion: entry.fromVersion,
172
+ toVersion: entry.toVersion,
173
+ restore: async () => {
174
+ await manager.restoreFromBackup(entry.id);
175
+ },
176
+ };
177
+ }
178
+
179
+ // ---------------------------------------------------------------------------
180
+ // Factory
181
+ // ---------------------------------------------------------------------------
182
+
183
+ export function createSafetyManager(
184
+ options: SafetyManagerOptions = {}
185
+ ): SafetyManager {
186
+ const projectRoot = options.projectRoot || process.cwd();
187
+ const skipBackup = options.skipBackup ?? false;
188
+ const dryRun = options.dryRun ?? false;
189
+
190
+ // Forward declaration so entryToBackup can reference `manager`.
191
+ const manager: SafetyManager = {
192
+ // ------------------------------------------------------------------
193
+ // createBackup
194
+ // ------------------------------------------------------------------
195
+ async createBackup(
196
+ files: string[],
197
+ description = 'Backup',
198
+ fromVersion = '',
199
+ toVersion = ''
200
+ ): Promise<UpdateBackup> {
201
+ if (skipBackup || dryRun) {
202
+ // Return a no-op backup object without touching disk.
203
+ const noop: UpdateBackup = {
204
+ id: `noop-${Date.now()}`,
205
+ timestamp: new Date(),
206
+ path: '',
207
+ files: [],
208
+ description,
209
+ fromVersion,
210
+ toVersion,
211
+ restore: async () => {},
212
+ };
213
+ return noop;
214
+ }
215
+
216
+ await ensureBackupDir(projectRoot);
217
+
218
+ const id = generateBackupId();
219
+ const backupPath = path.join(backupDir(projectRoot), id);
220
+ await fs.mkdir(backupPath, { recursive: true });
221
+
222
+ const backedUpFiles: string[] = [];
223
+
224
+ for (const filePath of files) {
225
+ if (!existsSync(filePath)) {
226
+ continue;
227
+ }
228
+
229
+ // Preserve the relative structure inside the backup directory.
230
+ // For absolute paths outside projectRoot use the full path stripped
231
+ // of its leading separator.
232
+ let relative: string;
233
+ try {
234
+ relative = path.relative(projectRoot, filePath);
235
+ // If the file is outside projectRoot, relative starts with '..'
236
+ if (relative.startsWith('..')) {
237
+ relative = filePath.replace(/^[/\\]/, '');
238
+ }
239
+ } catch {
240
+ relative = filePath.replace(/^[/\\]/, '');
241
+ }
242
+
243
+ const dest = path.join(backupPath, relative);
244
+ await fs.mkdir(path.dirname(dest), { recursive: true });
245
+ await fs.copyFile(filePath, dest);
246
+ backedUpFiles.push(filePath);
247
+ }
248
+
249
+ const entry: BackupIndexEntry = {
250
+ id,
251
+ timestamp: new Date().toISOString(),
252
+ path: backupPath,
253
+ files: backedUpFiles,
254
+ description,
255
+ fromVersion,
256
+ toVersion,
257
+ };
258
+
259
+ const index = await readIndex(projectRoot);
260
+ // Newest first.
261
+ index.unshift(entry);
262
+ await writeIndex(projectRoot, index);
263
+
264
+ return entryToBackup(entry, manager);
265
+ },
266
+
267
+ // ------------------------------------------------------------------
268
+ // startTransaction
269
+ // ------------------------------------------------------------------
270
+ startTransaction(label = 'transaction'): UpdateTransaction {
271
+ interface OpRecord {
272
+ op: TransactionOperation;
273
+ status: 'pending' | 'completed' | 'failed';
274
+ failReason?: string;
275
+ }
276
+
277
+ const ops = new Map<string, OpRecord>();
278
+ let committed = false;
279
+
280
+ const transaction: UpdateTransaction = {
281
+ recordOperation(op: TransactionOperation): void {
282
+ if (committed) {
283
+ throw new Error(
284
+ `Transaction "${label}" is already committed; cannot record new operations.`
285
+ );
286
+ }
287
+ ops.set(op.path, { op, status: 'pending' });
288
+ },
289
+
290
+ completeOperation(filePath: string): void {
291
+ const record = ops.get(filePath);
292
+ if (record) {
293
+ record.status = 'completed';
294
+ }
295
+ },
296
+
297
+ failOperation(filePath: string, reason: string): void {
298
+ const record = ops.get(filePath);
299
+ if (record) {
300
+ record.status = 'failed';
301
+ record.failReason = reason;
302
+ }
303
+ },
304
+
305
+ async commit(): Promise<boolean> {
306
+ committed = true;
307
+ // All operations are already written to disk by the caller; the
308
+ // transaction commit just seals the log.
309
+ return true;
310
+ },
311
+
312
+ async rollback(): Promise<void> {
313
+ // For each operation that has a backupRef, restore the file from
314
+ // the backup entry stored under projectRoot/.wundr-backup/{id}.
315
+ for (const [filePath, record] of ops.entries()) {
316
+ const { backupRef } = record.op;
317
+ if (!backupRef) {
318
+ continue;
319
+ }
320
+
321
+ try {
322
+ const index = await readIndex(projectRoot);
323
+ const entry = index.find(e => e.id === backupRef);
324
+ if (!entry) {
325
+ continue;
326
+ }
327
+
328
+ let relative: string;
329
+ try {
330
+ relative = path.relative(projectRoot, filePath);
331
+ if (relative.startsWith('..')) {
332
+ relative = filePath.replace(/^[/\\]/, '');
333
+ }
334
+ } catch {
335
+ relative = filePath.replace(/^[/\\]/, '');
336
+ }
337
+
338
+ const backupFilePath = path.join(entry.path, relative);
339
+ if (existsSync(backupFilePath)) {
340
+ await fs.mkdir(path.dirname(filePath), { recursive: true });
341
+ await fs.copyFile(backupFilePath, filePath);
342
+ }
343
+ } catch {
344
+ // Best-effort; continue with remaining files.
345
+ }
346
+ }
347
+ },
348
+ };
349
+
350
+ return transaction;
351
+ },
352
+
353
+ // ------------------------------------------------------------------
354
+ // listBackups
355
+ // ------------------------------------------------------------------
356
+ async listBackups(): Promise<UpdateBackup[]> {
357
+ await ensureBackupDir(projectRoot);
358
+ const index = await readIndex(projectRoot);
359
+ // Already sorted newest-first by how we write the index.
360
+ return index.map(entry => entryToBackup(entry, manager));
361
+ },
362
+
363
+ // ------------------------------------------------------------------
364
+ // getLatestBackup
365
+ // ------------------------------------------------------------------
366
+ async getLatestBackup(): Promise<UpdateBackup | null> {
367
+ await ensureBackupDir(projectRoot);
368
+ const index = await readIndex(projectRoot);
369
+ if (index.length === 0) {
370
+ return null;
371
+ }
372
+ return entryToBackup(index[0], manager);
373
+ },
374
+
375
+ // ------------------------------------------------------------------
376
+ // restoreFromBackup
377
+ // ------------------------------------------------------------------
378
+ async restoreFromBackup(
379
+ backupOrId: UpdateBackup | string
380
+ ): Promise<boolean> {
381
+ const id = typeof backupOrId === 'string' ? backupOrId : backupOrId.id;
382
+
383
+ const index = await readIndex(projectRoot);
384
+ const entry = index.find(e => e.id === id);
385
+ if (!entry) {
386
+ return false;
387
+ }
388
+
389
+ let allRestored = true;
390
+
391
+ for (const originalPath of entry.files) {
392
+ let relative: string;
393
+ try {
394
+ relative = path.relative(projectRoot, originalPath);
395
+ if (relative.startsWith('..')) {
396
+ relative = originalPath.replace(/^[/\\]/, '');
397
+ }
398
+ } catch {
399
+ relative = originalPath.replace(/^[/\\]/, '');
400
+ }
401
+
402
+ const backupFilePath = path.join(entry.path, relative);
403
+
404
+ if (!existsSync(backupFilePath)) {
405
+ allRestored = false;
406
+ continue;
407
+ }
408
+
409
+ try {
410
+ await fs.mkdir(path.dirname(originalPath), { recursive: true });
411
+ await fs.copyFile(backupFilePath, originalPath);
412
+ } catch {
413
+ allRestored = false;
414
+ }
415
+ }
416
+
417
+ return allRestored;
418
+ },
419
+
420
+ // ------------------------------------------------------------------
421
+ // deleteBackup
422
+ // ------------------------------------------------------------------
423
+ async deleteBackup(backupId: string): Promise<boolean> {
424
+ const index = await readIndex(projectRoot);
425
+ const entry = index.find(e => e.id === backupId);
426
+ if (!entry) {
427
+ return false;
428
+ }
429
+
430
+ try {
431
+ if (existsSync(entry.path)) {
432
+ await fs.rm(entry.path, { recursive: true, force: true });
433
+ }
434
+ const updated = index.filter(e => e.id !== backupId);
435
+ await writeIndex(projectRoot, updated);
436
+ return true;
437
+ } catch {
438
+ return false;
439
+ }
440
+ },
441
+
442
+ // ------------------------------------------------------------------
443
+ // Legacy: beginTransaction (alias)
444
+ // ------------------------------------------------------------------
445
+ beginTransaction(): UpdateTransaction {
446
+ return manager.startTransaction('transaction');
447
+ },
448
+ };
449
+
450
+ return manager;
451
+ }