@wundr.io/cli 1.0.0 → 1.0.3

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 (224) hide show
  1. package/README.md +696 -280
  2. package/dist/ai/ai-service.d.ts +2 -2
  3. package/dist/ai/ai-service.d.ts.map +1 -1
  4. package/dist/ai/ai-service.js +1 -1
  5. package/dist/ai/ai-service.js.map +1 -1
  6. package/dist/ai/claude-client.d.ts.map +1 -1
  7. package/dist/ai/claude-client.js +2 -1
  8. package/dist/ai/claude-client.js.map +1 -1
  9. package/dist/ai/conversation-manager.d.ts +2 -2
  10. package/dist/ai/conversation-manager.d.ts.map +1 -1
  11. package/dist/ai/conversation-manager.js +5 -3
  12. package/dist/ai/conversation-manager.js.map +1 -1
  13. package/dist/cli.d.ts.map +1 -1
  14. package/dist/cli.js +32 -13
  15. package/dist/cli.js.map +1 -1
  16. package/dist/commands/ai.d.ts +3 -3
  17. package/dist/commands/ai.d.ts.map +1 -1
  18. package/dist/commands/ai.js +2 -2
  19. package/dist/commands/ai.js.map +1 -1
  20. package/dist/commands/alignment.d.ts +78 -0
  21. package/dist/commands/alignment.d.ts.map +1 -0
  22. package/dist/commands/alignment.js +817 -0
  23. package/dist/commands/alignment.js.map +1 -0
  24. package/dist/commands/analyze-optimized.d.ts.map +1 -1
  25. package/dist/commands/analyze-optimized.js +9 -6
  26. package/dist/commands/analyze-optimized.js.map +1 -1
  27. package/dist/commands/analyze.d.ts +3 -3
  28. package/dist/commands/analyze.d.ts.map +1 -1
  29. package/dist/commands/analyze.js +2 -2
  30. package/dist/commands/analyze.js.map +1 -1
  31. package/dist/commands/batch.d.ts +3 -3
  32. package/dist/commands/batch.d.ts.map +1 -1
  33. package/dist/commands/batch.js +7 -7
  34. package/dist/commands/batch.js.map +1 -1
  35. package/dist/commands/chat.d.ts +3 -3
  36. package/dist/commands/chat.d.ts.map +1 -1
  37. package/dist/commands/chat.js +3 -3
  38. package/dist/commands/chat.js.map +1 -1
  39. package/dist/commands/claude-init.d.ts +1 -1
  40. package/dist/commands/claude-init.d.ts.map +1 -1
  41. package/dist/commands/claude-init.js +11 -7
  42. package/dist/commands/claude-init.js.map +1 -1
  43. package/dist/commands/claude-setup.d.ts +88 -1
  44. package/dist/commands/claude-setup.d.ts.map +1 -1
  45. package/dist/commands/claude-setup.js +549 -46
  46. package/dist/commands/claude-setup.js.map +1 -1
  47. package/dist/commands/computer-setup-commands.d.ts +17 -3
  48. package/dist/commands/computer-setup-commands.d.ts.map +1 -1
  49. package/dist/commands/computer-setup-commands.js +145 -3
  50. package/dist/commands/computer-setup-commands.js.map +1 -1
  51. package/dist/commands/computer-setup.d.ts.map +1 -1
  52. package/dist/commands/computer-setup.js +372 -4
  53. package/dist/commands/computer-setup.js.map +1 -1
  54. package/dist/commands/create-command.d.ts.map +1 -1
  55. package/dist/commands/create-command.js +3 -3
  56. package/dist/commands/create-command.js.map +1 -1
  57. package/dist/commands/create.d.ts +3 -3
  58. package/dist/commands/create.d.ts.map +1 -1
  59. package/dist/commands/create.js +3 -3
  60. package/dist/commands/create.js.map +1 -1
  61. package/dist/commands/dashboard.d.ts +3 -3
  62. package/dist/commands/dashboard.d.ts.map +1 -1
  63. package/dist/commands/dashboard.js +4 -3
  64. package/dist/commands/dashboard.js.map +1 -1
  65. package/dist/commands/govern.d.ts +3 -3
  66. package/dist/commands/govern.d.ts.map +1 -1
  67. package/dist/commands/govern.js +4 -3
  68. package/dist/commands/govern.js.map +1 -1
  69. package/dist/commands/governance.d.ts +17 -0
  70. package/dist/commands/governance.d.ts.map +1 -0
  71. package/dist/commands/governance.js +703 -0
  72. package/dist/commands/governance.js.map +1 -0
  73. package/dist/commands/guardian.d.ts +20 -0
  74. package/dist/commands/guardian.d.ts.map +1 -0
  75. package/dist/commands/guardian.js +597 -0
  76. package/dist/commands/guardian.js.map +1 -0
  77. package/dist/commands/init.d.ts +7 -3
  78. package/dist/commands/init.d.ts.map +1 -1
  79. package/dist/commands/init.js +71 -5
  80. package/dist/commands/init.js.map +1 -1
  81. package/dist/commands/performance-optimizer.d.ts +2 -2
  82. package/dist/commands/performance-optimizer.d.ts.map +1 -1
  83. package/dist/commands/performance-optimizer.js +8 -7
  84. package/dist/commands/performance-optimizer.js.map +1 -1
  85. package/dist/commands/plugins.d.ts +3 -3
  86. package/dist/commands/plugins.d.ts.map +1 -1
  87. package/dist/commands/plugins.js +2 -2
  88. package/dist/commands/plugins.js.map +1 -1
  89. package/dist/commands/rag.d.ts +7 -0
  90. package/dist/commands/rag.d.ts.map +1 -0
  91. package/dist/commands/rag.js +748 -0
  92. package/dist/commands/rag.js.map +1 -0
  93. package/dist/commands/session.d.ts +41 -0
  94. package/dist/commands/session.d.ts.map +1 -0
  95. package/dist/commands/session.js +441 -0
  96. package/dist/commands/session.js.map +1 -0
  97. package/dist/commands/setup.d.ts +3 -3
  98. package/dist/commands/setup.d.ts.map +1 -1
  99. package/dist/commands/setup.js +1 -3
  100. package/dist/commands/setup.js.map +1 -1
  101. package/dist/commands/test-init.d.ts.map +1 -1
  102. package/dist/commands/test-init.js +2 -2
  103. package/dist/commands/test-init.js.map +1 -1
  104. package/dist/commands/test.d.ts.map +1 -1
  105. package/dist/commands/test.js +2 -2
  106. package/dist/commands/test.js.map +1 -1
  107. package/dist/commands/vp.d.ts +7 -0
  108. package/dist/commands/vp.d.ts.map +1 -0
  109. package/dist/commands/vp.js +571 -0
  110. package/dist/commands/vp.js.map +1 -0
  111. package/dist/commands/watch.d.ts +3 -3
  112. package/dist/commands/watch.d.ts.map +1 -1
  113. package/dist/commands/watch.js +10 -7
  114. package/dist/commands/watch.js.map +1 -1
  115. package/dist/commands/worktree.d.ts +63 -0
  116. package/dist/commands/worktree.d.ts.map +1 -0
  117. package/dist/commands/worktree.js +774 -0
  118. package/dist/commands/worktree.js.map +1 -0
  119. package/dist/context/context-manager.d.ts +1 -1
  120. package/dist/context/context-manager.d.ts.map +1 -1
  121. package/dist/context/context-manager.js +1 -1
  122. package/dist/context/context-manager.js.map +1 -1
  123. package/dist/context/session-manager.d.ts +2 -2
  124. package/dist/context/session-manager.d.ts.map +1 -1
  125. package/dist/context/session-manager.js +9 -5
  126. package/dist/context/session-manager.js.map +1 -1
  127. package/dist/index.d.ts.map +1 -1
  128. package/dist/index.js +1 -1
  129. package/dist/index.js.map +1 -1
  130. package/dist/interactive/interactive-mode.d.ts +2 -2
  131. package/dist/interactive/interactive-mode.d.ts.map +1 -1
  132. package/dist/interactive/interactive-mode.js +6 -4
  133. package/dist/interactive/interactive-mode.js.map +1 -1
  134. package/dist/nlp/command-mapper.d.ts +1 -1
  135. package/dist/nlp/command-mapper.d.ts.map +1 -1
  136. package/dist/nlp/command-mapper.js +3 -2
  137. package/dist/nlp/command-mapper.js.map +1 -1
  138. package/dist/nlp/command-parser.d.ts +1 -1
  139. package/dist/nlp/command-parser.d.ts.map +1 -1
  140. package/dist/nlp/command-parser.js +2 -1
  141. package/dist/nlp/command-parser.js.map +1 -1
  142. package/dist/nlp/intent-parser.d.ts +1 -1
  143. package/dist/nlp/intent-parser.d.ts.map +1 -1
  144. package/dist/nlp/intent-parser.js +14 -9
  145. package/dist/nlp/intent-parser.js.map +1 -1
  146. package/dist/plugins/plugin-manager.d.ts +2 -2
  147. package/dist/plugins/plugin-manager.d.ts.map +1 -1
  148. package/dist/plugins/plugin-manager.js +3 -3
  149. package/dist/plugins/plugin-manager.js.map +1 -1
  150. package/dist/types/index.d.ts +1 -1
  151. package/dist/types/index.d.ts.map +1 -1
  152. package/dist/utils/backup-rollback-manager.d.ts +72 -0
  153. package/dist/utils/backup-rollback-manager.d.ts.map +1 -0
  154. package/dist/utils/backup-rollback-manager.js +289 -0
  155. package/dist/utils/backup-rollback-manager.js.map +1 -0
  156. package/dist/utils/claude-config-installer.d.ts +94 -0
  157. package/dist/utils/claude-config-installer.d.ts.map +1 -0
  158. package/dist/utils/claude-config-installer.js +628 -0
  159. package/dist/utils/claude-config-installer.js.map +1 -0
  160. package/dist/utils/config-manager.d.ts +1 -1
  161. package/dist/utils/config-manager.d.ts.map +1 -1
  162. package/dist/utils/config-manager.js +5 -5
  163. package/dist/utils/config-manager.js.map +1 -1
  164. package/dist/utils/error-handler.d.ts +1 -1
  165. package/dist/utils/error-handler.d.ts.map +1 -1
  166. package/dist/utils/error-handler.js.map +1 -1
  167. package/dist/utils/logger.d.ts +1 -1
  168. package/dist/utils/logger.d.ts.map +1 -1
  169. package/dist/utils/logger.js +22 -11
  170. package/dist/utils/logger.js.map +1 -1
  171. package/package.json +26 -7
  172. package/src/ai/ai-service.ts +22 -19
  173. package/src/ai/claude-client.ts +20 -16
  174. package/src/ai/conversation-manager.ts +37 -30
  175. package/src/cli.ts +42 -13
  176. package/src/commands/ai.ts +59 -57
  177. package/src/commands/alignment.ts +1212 -0
  178. package/src/commands/analyze-optimized.ts +70 -62
  179. package/src/commands/analyze.ts +22 -20
  180. package/src/commands/batch.ts +41 -38
  181. package/src/commands/chat.ts +37 -34
  182. package/src/commands/claude-init.ts +38 -30
  183. package/src/commands/claude-setup.ts +692 -97
  184. package/src/commands/computer-setup-commands.ts +203 -37
  185. package/src/commands/computer-setup.ts +474 -4
  186. package/src/commands/create-command.ts +7 -7
  187. package/src/commands/create.ts +36 -33
  188. package/src/commands/dashboard.ts +33 -28
  189. package/src/commands/govern.ts +34 -29
  190. package/src/commands/governance.ts +1005 -0
  191. package/src/commands/guardian.ts +887 -0
  192. package/src/commands/init.ts +112 -22
  193. package/src/commands/performance-optimizer.ts +48 -42
  194. package/src/commands/plugins.ts +35 -32
  195. package/src/commands/project-update.ts +1053 -0
  196. package/src/commands/rag.ts +904 -0
  197. package/src/commands/session.ts +631 -0
  198. package/src/commands/setup.ts +35 -31
  199. package/src/commands/test-init.ts +6 -5
  200. package/src/commands/test.ts +7 -6
  201. package/src/commands/vp.ts +762 -0
  202. package/src/commands/watch.ts +44 -33
  203. package/src/commands/worktree.ts +1057 -0
  204. package/src/context/context-manager.ts +15 -12
  205. package/src/context/session-manager.ts +51 -40
  206. package/src/index.ts +7 -6
  207. package/src/interactive/interactive-mode.ts +25 -18
  208. package/src/lib/conflict-resolution.ts +28 -0
  209. package/src/lib/merge-strategy.ts +28 -0
  210. package/src/lib/safety-mechanisms.ts +47 -0
  211. package/src/lib/state-detection.ts +28 -0
  212. package/src/nlp/command-mapper.ts +35 -30
  213. package/src/nlp/command-parser.ts +20 -17
  214. package/src/nlp/intent-classifier.ts +7 -7
  215. package/src/nlp/intent-parser.ts +61 -49
  216. package/src/plugins/plugin-manager.ts +27 -23
  217. package/src/tests/computer-setup-integration.test.ts +439 -0
  218. package/src/types/index.ts +1 -1
  219. package/src/types/modules.d.ts +1 -0
  220. package/src/utils/backup-rollback-manager.ts +363 -0
  221. package/src/utils/claude-config-installer.ts +734 -0
  222. package/src/utils/config-manager.ts +12 -9
  223. package/src/utils/error-handler.ts +5 -3
  224. package/src/utils/logger.ts +35 -12
@@ -0,0 +1,363 @@
1
+ /**
2
+ * Backup and Rollback Manager
3
+ * Handles backup creation and restoration for configuration files
4
+ */
5
+
6
+ import { existsSync } from 'fs';
7
+ import * as fs from 'fs/promises';
8
+ import * as path from 'path';
9
+
10
+ import chalk from 'chalk';
11
+
12
+ import { logger } from './logger';
13
+
14
+ export interface BackupMetadata {
15
+ timestamp: string;
16
+ backupId: string;
17
+ files: BackupFile[];
18
+ reason: string;
19
+ success: boolean;
20
+ }
21
+
22
+ export interface BackupFile {
23
+ originalPath: string;
24
+ backupPath: string;
25
+ size: number;
26
+ checksum?: string;
27
+ }
28
+
29
+ export interface RollbackOptions {
30
+ backupId?: string;
31
+ dryRun?: boolean;
32
+ verbose?: boolean;
33
+ }
34
+
35
+ export class BackupRollbackManager {
36
+ private backupDir: string;
37
+ private metadataFile: string;
38
+ private homeDir: string;
39
+
40
+ constructor(backupDir?: string) {
41
+ this.homeDir = process.env.HOME || process.env.USERPROFILE || '';
42
+ this.backupDir =
43
+ backupDir || path.join(this.homeDir, '.wundr', 'backups');
44
+ this.metadataFile = path.join(this.backupDir, 'metadata.json');
45
+ }
46
+
47
+ /**
48
+ * Initialize backup directory structure
49
+ */
50
+ async initialize(): Promise<void> {
51
+ try {
52
+ await fs.mkdir(this.backupDir, { recursive: true });
53
+
54
+ if (!existsSync(this.metadataFile)) {
55
+ await fs.writeFile(this.metadataFile, JSON.stringify([], null, 2));
56
+ }
57
+
58
+ logger.info('Backup manager initialized', { backupDir: this.backupDir });
59
+ } catch (error) {
60
+ logger.error('Failed to initialize backup manager', error);
61
+ throw error;
62
+ }
63
+ }
64
+
65
+ /**
66
+ * Create backup of specified files
67
+ */
68
+ async createBackup(
69
+ files: string[],
70
+ reason: string = 'Manual backup',
71
+ ): Promise<BackupMetadata> {
72
+ const backupId = this.generateBackupId();
73
+ const timestamp = new Date().toISOString();
74
+ const backupPath = path.join(this.backupDir, backupId);
75
+
76
+ logger.info('Creating backup', { backupId, files: files.length, reason });
77
+
78
+ try {
79
+ await fs.mkdir(backupPath, { recursive: true });
80
+
81
+ const backupFiles: BackupFile[] = [];
82
+
83
+ for (const filePath of files) {
84
+ const expandedPath = this.expandPath(filePath);
85
+
86
+ if (!existsSync(expandedPath)) {
87
+ logger.warn('File not found, skipping', { file: expandedPath });
88
+ continue;
89
+ }
90
+
91
+ const stats = await fs.stat(expandedPath);
92
+ const relativePath = this.getRelativePath(expandedPath);
93
+ const backupFilePath = path.join(backupPath, relativePath);
94
+
95
+ // Create directory structure
96
+ await fs.mkdir(path.dirname(backupFilePath), { recursive: true });
97
+
98
+ // Copy file
99
+ await fs.copyFile(expandedPath, backupFilePath);
100
+
101
+ backupFiles.push({
102
+ originalPath: expandedPath,
103
+ backupPath: backupFilePath,
104
+ size: stats.size,
105
+ });
106
+
107
+ logger.info('Backed up file', {
108
+ original: expandedPath,
109
+ backup: backupFilePath,
110
+ });
111
+ }
112
+
113
+ const metadata: BackupMetadata = {
114
+ timestamp,
115
+ backupId,
116
+ files: backupFiles,
117
+ reason,
118
+ success: true,
119
+ };
120
+
121
+ await this.saveMetadata(metadata);
122
+
123
+ logger.info('Backup created successfully', {
124
+ backupId,
125
+ filesBackedUp: backupFiles.length,
126
+ });
127
+
128
+ return metadata;
129
+ } catch (error) {
130
+ logger.error('Backup failed', error);
131
+
132
+ const metadata: BackupMetadata = {
133
+ timestamp,
134
+ backupId,
135
+ files: [],
136
+ reason,
137
+ success: false,
138
+ };
139
+
140
+ await this.saveMetadata(metadata);
141
+ throw error;
142
+ }
143
+ }
144
+
145
+ /**
146
+ * Restore from backup
147
+ */
148
+ async rollback(options: RollbackOptions = {}): Promise<boolean> {
149
+ const { backupId, dryRun = false, verbose = false } = options;
150
+
151
+ try {
152
+ const metadata = backupId
153
+ ? await this.getBackupMetadata(backupId)
154
+ : await this.getLatestBackup();
155
+
156
+ if (!metadata) {
157
+ logger.error('No backup found to restore');
158
+ return false;
159
+ }
160
+
161
+ logger.info('Rolling back', {
162
+ backupId: metadata.backupId,
163
+ dryRun,
164
+ files: metadata.files.length,
165
+ });
166
+
167
+ if (dryRun) {
168
+ console.log(chalk.yellow('\nšŸ” DRY RUN - No files will be modified\n'));
169
+ console.log(chalk.cyan('Files that would be restored:'));
170
+ metadata.files.forEach(file => {
171
+ console.log(chalk.white(` ${file.originalPath}`));
172
+ if (verbose) {
173
+ console.log(chalk.gray(` ← ${file.backupPath}`));
174
+ }
175
+ });
176
+ return true;
177
+ }
178
+
179
+ const restoredFiles: string[] = [];
180
+ const failedFiles: string[] = [];
181
+
182
+ for (const file of metadata.files) {
183
+ try {
184
+ // Create directory structure
185
+ await fs.mkdir(path.dirname(file.originalPath), {
186
+ recursive: true,
187
+ });
188
+
189
+ // Restore file
190
+ await fs.copyFile(file.backupPath, file.originalPath);
191
+ restoredFiles.push(file.originalPath);
192
+
193
+ logger.info('Restored file', { file: file.originalPath });
194
+ } catch (error) {
195
+ logger.error('Failed to restore file', {
196
+ file: file.originalPath,
197
+ error,
198
+ });
199
+ failedFiles.push(file.originalPath);
200
+ }
201
+ }
202
+
203
+ console.log(chalk.green(`\nāœ… Restored ${restoredFiles.length} files`));
204
+
205
+ if (failedFiles.length > 0) {
206
+ console.log(chalk.red(`āŒ Failed to restore ${failedFiles.length} files`));
207
+ failedFiles.forEach(file => {
208
+ console.log(chalk.red(` - ${file}`));
209
+ });
210
+ return false;
211
+ }
212
+
213
+ return true;
214
+ } catch (error) {
215
+ logger.error('Rollback failed', error);
216
+ return false;
217
+ }
218
+ }
219
+
220
+ /**
221
+ * List all backups
222
+ */
223
+ async listBackups(): Promise<BackupMetadata[]> {
224
+ try {
225
+ const content = await fs.readFile(this.metadataFile, 'utf-8');
226
+ const backups = JSON.parse(content) as BackupMetadata[];
227
+ return backups.sort((a, b) =>
228
+ new Date(b.timestamp).getTime() - new Date(a.timestamp).getTime(),
229
+ );
230
+ } catch (error) {
231
+ logger.error('Failed to list backups', error);
232
+ return [];
233
+ }
234
+ }
235
+
236
+ /**
237
+ * Get specific backup metadata
238
+ */
239
+ async getBackupMetadata(backupId: string): Promise<BackupMetadata | null> {
240
+ const backups = await this.listBackups();
241
+ return backups.find(b => b.backupId === backupId) || null;
242
+ }
243
+
244
+ /**
245
+ * Get latest successful backup
246
+ */
247
+ async getLatestBackup(): Promise<BackupMetadata | null> {
248
+ const backups = await this.listBackups();
249
+ return backups.find(b => b.success) || null;
250
+ }
251
+
252
+ /**
253
+ * Delete old backups
254
+ */
255
+ async cleanupOldBackups(retainCount: number = 5): Promise<void> {
256
+ const backups = await this.listBackups();
257
+
258
+ if (backups.length <= retainCount) {
259
+ logger.info('No backups to clean up', {
260
+ current: backups.length,
261
+ retain: retainCount,
262
+ });
263
+ return;
264
+ }
265
+
266
+ const toDelete = backups.slice(retainCount);
267
+
268
+ logger.info('Cleaning up old backups', { count: toDelete.length });
269
+
270
+ for (const backup of toDelete) {
271
+ try {
272
+ const backupPath = path.join(this.backupDir, backup.backupId);
273
+ await fs.rm(backupPath, { recursive: true, force: true });
274
+ logger.info('Deleted backup', { backupId: backup.backupId });
275
+ } catch (error) {
276
+ logger.error('Failed to delete backup', {
277
+ backupId: backup.backupId,
278
+ error,
279
+ });
280
+ }
281
+ }
282
+
283
+ // Update metadata
284
+ const remaining = backups.slice(0, retainCount);
285
+ await fs.writeFile(this.metadataFile, JSON.stringify(remaining, null, 2));
286
+ }
287
+
288
+ /**
289
+ * Verify backup integrity
290
+ */
291
+ async verifyBackup(backupId: string): Promise<boolean> {
292
+ const metadata = await this.getBackupMetadata(backupId);
293
+
294
+ if (!metadata) {
295
+ logger.error('Backup not found', { backupId });
296
+ return false;
297
+ }
298
+
299
+ let valid = true;
300
+
301
+ for (const file of metadata.files) {
302
+ if (!existsSync(file.backupPath)) {
303
+ logger.error('Backup file missing', { file: file.backupPath });
304
+ valid = false;
305
+ }
306
+ }
307
+
308
+ return valid;
309
+ }
310
+
311
+ /**
312
+ * Private helper methods
313
+ */
314
+
315
+ private generateBackupId(): string {
316
+ const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
317
+ return `backup-${timestamp}`;
318
+ }
319
+
320
+ private expandPath(filePath: string): string {
321
+ if (filePath.startsWith('~/')) {
322
+ return path.join(this.homeDir, filePath.slice(2));
323
+ }
324
+ return filePath;
325
+ }
326
+
327
+ private getRelativePath(absolutePath: string): string {
328
+ // Create relative path from home directory
329
+ if (absolutePath.startsWith(this.homeDir)) {
330
+ return path.relative(this.homeDir, absolutePath);
331
+ }
332
+ // For absolute paths outside home, use full path structure
333
+ return absolutePath.replace(/^\//, '');
334
+ }
335
+
336
+ private async saveMetadata(metadata: BackupMetadata): Promise<void> {
337
+ const backups = await this.listBackups();
338
+ backups.unshift(metadata);
339
+ await fs.writeFile(this.metadataFile, JSON.stringify(backups, null, 2));
340
+ }
341
+
342
+ /**
343
+ * Display backup information
344
+ */
345
+ displayBackupInfo(metadata: BackupMetadata): void {
346
+ console.log(chalk.cyan('\nšŸ“¦ Backup Information\n'));
347
+ console.log(chalk.white('Backup ID:'), chalk.green(metadata.backupId));
348
+ console.log(chalk.white('Timestamp:'), chalk.gray(metadata.timestamp));
349
+ console.log(chalk.white('Reason:'), chalk.gray(metadata.reason));
350
+ console.log(chalk.white('Status:'),
351
+ metadata.success ? chalk.green('Success') : chalk.red('Failed'),
352
+ );
353
+ console.log(chalk.white('Files:'), chalk.cyan(metadata.files.length));
354
+
355
+ if (metadata.files.length > 0) {
356
+ console.log(chalk.cyan('\nBacked up files:'));
357
+ metadata.files.forEach(file => {
358
+ const size = (file.size / 1024).toFixed(2);
359
+ console.log(chalk.gray(` • ${file.originalPath} (${size} KB)`));
360
+ });
361
+ }
362
+ }
363
+ }