driftdetect 0.7.1 → 0.8.0

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 (85) hide show
  1. package/dist/bin/drift.d.ts +11 -0
  2. package/dist/bin/drift.js +26 -1
  3. package/dist/bin/drift.js.map +1 -1
  4. package/dist/commands/approve.d.ts +2 -0
  5. package/dist/commands/approve.d.ts.map +1 -1
  6. package/dist/commands/approve.js +92 -1
  7. package/dist/commands/approve.js.map +1 -1
  8. package/dist/commands/boundaries.d.ts +20 -0
  9. package/dist/commands/callgraph.d.ts.map +1 -1
  10. package/dist/commands/callgraph.js +42 -52
  11. package/dist/commands/callgraph.js.map +1 -1
  12. package/dist/commands/context.d.ts +19 -0
  13. package/dist/commands/context.d.ts.map +1 -0
  14. package/dist/commands/context.js +301 -0
  15. package/dist/commands/context.js.map +1 -0
  16. package/dist/commands/cpp.d.ts +22 -0
  17. package/dist/commands/cpp.d.ts.map +1 -0
  18. package/dist/commands/cpp.js +485 -0
  19. package/dist/commands/cpp.js.map +1 -0
  20. package/dist/commands/dashboard.d.ts +16 -0
  21. package/dist/commands/dashboard.d.ts.map +1 -1
  22. package/dist/commands/dashboard.js +3 -1
  23. package/dist/commands/dashboard.js.map +1 -1
  24. package/dist/commands/dna/export.d.ts +6 -0
  25. package/dist/commands/dna/gene.d.ts +6 -0
  26. package/dist/commands/dna/index.d.ts +8 -0
  27. package/dist/commands/dna/mutations.d.ts +6 -0
  28. package/dist/commands/dna/playbook.d.ts +6 -0
  29. package/dist/commands/dna/scan.d.ts +6 -0
  30. package/dist/commands/dna/status.d.ts +6 -0
  31. package/dist/commands/gate.d.ts +33 -0
  32. package/dist/commands/gate.d.ts.map +1 -0
  33. package/dist/commands/gate.js +199 -0
  34. package/dist/commands/gate.js.map +1 -0
  35. package/dist/commands/ignore.d.ts.map +1 -1
  36. package/dist/commands/ignore.js +36 -0
  37. package/dist/commands/ignore.js.map +1 -1
  38. package/dist/commands/index.d.ts +6 -0
  39. package/dist/commands/index.d.ts.map +1 -1
  40. package/dist/commands/index.js +12 -0
  41. package/dist/commands/index.js.map +1 -1
  42. package/dist/commands/init.d.ts +19 -0
  43. package/dist/commands/init.d.ts.map +1 -1
  44. package/dist/commands/init.js +41 -1
  45. package/dist/commands/init.js.map +1 -1
  46. package/dist/commands/license.d.ts +12 -0
  47. package/dist/commands/license.d.ts.map +1 -0
  48. package/dist/commands/license.js +133 -0
  49. package/dist/commands/license.js.map +1 -0
  50. package/dist/commands/parser.d.ts +20 -0
  51. package/dist/commands/rust.d.ts +21 -0
  52. package/dist/commands/rust.d.ts.map +1 -0
  53. package/dist/commands/rust.js +526 -0
  54. package/dist/commands/rust.js.map +1 -0
  55. package/dist/commands/scan.d.ts.map +1 -1
  56. package/dist/commands/scan.js +69 -1
  57. package/dist/commands/scan.js.map +1 -1
  58. package/dist/commands/simulate.d.ts.map +1 -1
  59. package/dist/commands/simulate.js +22 -1
  60. package/dist/commands/simulate.js.map +1 -1
  61. package/dist/commands/telemetry.d.ts +9 -0
  62. package/dist/commands/telemetry.d.ts.map +1 -0
  63. package/dist/commands/telemetry.js +289 -0
  64. package/dist/commands/telemetry.js.map +1 -0
  65. package/dist/commands/trends.d.ts +12 -0
  66. package/dist/commands/watch.d.ts +13 -0
  67. package/dist/git/hooks.d.ts +108 -0
  68. package/dist/git/index.d.ts +6 -0
  69. package/dist/git/staged-files.d.ts +41 -0
  70. package/dist/reporters/github-reporter.d.ts +13 -0
  71. package/dist/reporters/gitlab-reporter.d.ts +16 -0
  72. package/dist/reporters/index.d.ts +9 -0
  73. package/dist/reporters/json-reporter.d.ts +13 -0
  74. package/dist/reporters/text-reporter.d.ts +13 -0
  75. package/dist/reporters/types.d.ts +42 -0
  76. package/dist/services/boundary-scanner.d.ts +21 -0
  77. package/dist/services/contract-scanner.d.ts +35 -0
  78. package/dist/types/index.d.ts +24 -0
  79. package/dist/ui/index.d.ts +11 -0
  80. package/dist/ui/progress.d.ts +115 -0
  81. package/dist/ui/prompts.d.ts +91 -0
  82. package/dist/ui/spinner.d.ts +109 -0
  83. package/dist/ui/table.d.ts +118 -0
  84. package/package.json +14 -14
  85. package/LICENSE +0 -21
@@ -0,0 +1,526 @@
1
+ /**
2
+ * Rust Command - drift rust
3
+ *
4
+ * Analyze Rust projects: routes, error handling, traits, data access.
5
+ *
6
+ * @requirements Rust Language Support
7
+ */
8
+ import { Command } from 'commander';
9
+ import chalk from 'chalk';
10
+ import { createRustAnalyzer } from 'driftdetect-core';
11
+ import { createSpinner } from '../ui/spinner.js';
12
+ /**
13
+ * Create the Rust command
14
+ */
15
+ export function createRustCommand() {
16
+ const rust = new Command('rust')
17
+ .description('Rust language analysis commands');
18
+ // drift rust routes
19
+ rust
20
+ .command('routes [path]')
21
+ .description('List all HTTP routes (Actix, Axum, Rocket, Warp)')
22
+ .option('-f, --format <format>', 'Output format: text, json', 'text')
23
+ .option('-v, --verbose', 'Enable verbose output')
24
+ .option('--framework <framework>', 'Filter by framework')
25
+ .action(async (targetPath, options) => {
26
+ await routesAction(targetPath, options);
27
+ });
28
+ // drift rust errors
29
+ rust
30
+ .command('errors [path]')
31
+ .description('Analyze error handling patterns (Result, thiserror, anyhow)')
32
+ .option('-f, --format <format>', 'Output format: text, json', 'text')
33
+ .option('-v, --verbose', 'Enable verbose output')
34
+ .action(async (targetPath, options) => {
35
+ await errorsAction(targetPath, options);
36
+ });
37
+ // drift rust traits
38
+ rust
39
+ .command('traits [path]')
40
+ .description('List traits and their implementations')
41
+ .option('-f, --format <format>', 'Output format: text, json', 'text')
42
+ .option('-v, --verbose', 'Enable verbose output')
43
+ .action(async (targetPath, options) => {
44
+ await traitsAction(targetPath, options);
45
+ });
46
+ // drift rust data-access
47
+ rust
48
+ .command('data-access [path]')
49
+ .description('Analyze database access patterns (SQLx, Diesel, SeaORM)')
50
+ .option('-f, --format <format>', 'Output format: text, json', 'text')
51
+ .option('-v, --verbose', 'Enable verbose output')
52
+ .action(async (targetPath, options) => {
53
+ await dataAccessAction(targetPath, options);
54
+ });
55
+ // drift rust async
56
+ rust
57
+ .command('async [path]')
58
+ .description('Analyze async patterns and runtime usage')
59
+ .option('-f, --format <format>', 'Output format: text, json', 'text')
60
+ .option('-v, --verbose', 'Enable verbose output')
61
+ .action(async (targetPath, options) => {
62
+ await asyncAction(targetPath, options);
63
+ });
64
+ // drift rust status
65
+ rust
66
+ .command('status [path]')
67
+ .description('Show Rust project analysis summary')
68
+ .option('-f, --format <format>', 'Output format: text, json', 'text')
69
+ .option('-v, --verbose', 'Enable verbose output')
70
+ .action(async (targetPath, options) => {
71
+ await statusAction(targetPath, options);
72
+ });
73
+ return rust;
74
+ }
75
+ /**
76
+ * Routes subcommand
77
+ */
78
+ async function routesAction(targetPath, options) {
79
+ const rootDir = targetPath ?? process.cwd();
80
+ const format = options.format ?? 'text';
81
+ const isTextFormat = format === 'text';
82
+ const spinner = isTextFormat ? createSpinner('Analyzing Rust routes...') : null;
83
+ spinner?.start();
84
+ try {
85
+ const analyzer = createRustAnalyzer({ rootDir, verbose: options.verbose ?? false });
86
+ const result = await analyzer.analyzeRoutes();
87
+ spinner?.stop();
88
+ // Filter by framework if specified
89
+ let routes = result.routes;
90
+ if (options.framework) {
91
+ routes = routes.filter((r) => r.framework === options.framework);
92
+ }
93
+ // JSON output
94
+ if (format === 'json') {
95
+ console.log(JSON.stringify({
96
+ total: routes.length,
97
+ byFramework: result.byFramework,
98
+ routes,
99
+ }, null, 2));
100
+ return;
101
+ }
102
+ // Text output
103
+ console.log();
104
+ console.log(chalk.bold('🦀 Rust HTTP Routes'));
105
+ console.log(chalk.gray('─'.repeat(60)));
106
+ console.log();
107
+ if (routes.length === 0) {
108
+ console.log(chalk.gray('No routes found'));
109
+ console.log();
110
+ return;
111
+ }
112
+ // Group by framework
113
+ const byFramework = new Map();
114
+ for (const route of routes) {
115
+ const existing = byFramework.get(route.framework) ?? [];
116
+ existing.push(route);
117
+ byFramework.set(route.framework, existing);
118
+ }
119
+ for (const [framework, frameworkRoutes] of byFramework) {
120
+ console.log(chalk.bold(`${framework} (${frameworkRoutes.length} routes)`));
121
+ for (const route of frameworkRoutes) {
122
+ const methodColor = getMethodColor(route.method);
123
+ console.log(` ${methodColor(route.method.padEnd(7))} ${route.path}`);
124
+ console.log(chalk.gray(` → ${route.handler} (${route.file}:${route.line})`));
125
+ }
126
+ console.log();
127
+ }
128
+ console.log(`Total: ${chalk.cyan(routes.length)} routes`);
129
+ console.log();
130
+ }
131
+ catch (error) {
132
+ spinner?.stop();
133
+ if (format === 'json') {
134
+ console.log(JSON.stringify({ error: String(error) }));
135
+ }
136
+ else {
137
+ console.log(chalk.red(`\n❌ Error: ${error}`));
138
+ }
139
+ }
140
+ }
141
+ /**
142
+ * Errors subcommand
143
+ */
144
+ async function errorsAction(targetPath, options) {
145
+ const rootDir = targetPath ?? process.cwd();
146
+ const format = options.format ?? 'text';
147
+ const isTextFormat = format === 'text';
148
+ const spinner = isTextFormat ? createSpinner('Analyzing error handling...') : null;
149
+ spinner?.start();
150
+ try {
151
+ const analyzer = createRustAnalyzer({ rootDir, verbose: options.verbose ?? false });
152
+ const result = await analyzer.analyzeErrorHandling();
153
+ spinner?.stop();
154
+ // JSON output
155
+ if (format === 'json') {
156
+ console.log(JSON.stringify(result, null, 2));
157
+ return;
158
+ }
159
+ // Text output
160
+ console.log();
161
+ console.log(chalk.bold('⚠️ Rust Error Handling Analysis'));
162
+ console.log(chalk.gray('─'.repeat(60)));
163
+ console.log();
164
+ console.log(`Result Types: ${chalk.cyan(result.stats.resultTypes)}`);
165
+ console.log(`Custom Errors: ${chalk.green(result.stats.customErrors)}`);
166
+ console.log(`thiserror Derives: ${chalk.blue(result.stats.thiserrorDerives)}`);
167
+ console.log(`anyhow Usage: ${chalk.magenta(result.stats.anyhowUsage)}`);
168
+ console.log(`Unwrap Calls: ${chalk.yellow(result.stats.unwrapCalls)}`);
169
+ console.log(`Expect Calls: ${chalk.yellow(result.stats.expectCalls)}`);
170
+ console.log();
171
+ // Pattern breakdown
172
+ const patternCounts = {
173
+ propagated: result.patterns.filter((p) => p.type === 'propagated').length,
174
+ wrapped: result.patterns.filter((p) => p.type === 'wrapped').length,
175
+ logged: result.patterns.filter((p) => p.type === 'logged').length,
176
+ ignored: result.patterns.filter((p) => p.type === 'ignored').length,
177
+ };
178
+ console.log(chalk.bold('Pattern Breakdown:'));
179
+ console.log(` Propagated (?): ${chalk.cyan(patternCounts.propagated)}`);
180
+ console.log(` Wrapped (.map_err): ${chalk.green(patternCounts.wrapped)}`);
181
+ console.log(` Logged: ${chalk.blue(patternCounts.logged)}`);
182
+ console.log(` Ignored: ${chalk.yellow(patternCounts.ignored)}`);
183
+ console.log();
184
+ // Issues
185
+ if (result.issues.length > 0) {
186
+ console.log(chalk.bold('Issues:'));
187
+ for (const issue of result.issues.slice(0, 10)) {
188
+ console.log(` ${chalk.yellow('⚠')} ${issue.file}:${issue.line}`);
189
+ console.log(chalk.gray(` ${issue.message}`));
190
+ if (issue.suggestion) {
191
+ console.log(chalk.gray(` → ${issue.suggestion}`));
192
+ }
193
+ }
194
+ if (result.issues.length > 10) {
195
+ console.log(chalk.gray(` ... and ${result.issues.length - 10} more`));
196
+ }
197
+ console.log();
198
+ }
199
+ // Custom error types
200
+ if (result.customErrors.length > 0 && options.verbose) {
201
+ console.log(chalk.bold('Custom Error Types:'));
202
+ for (const err of result.customErrors.slice(0, 10)) {
203
+ console.log(` ${chalk.blue(err.name)}`);
204
+ console.log(chalk.gray(` ${err.file}:${err.line}`));
205
+ }
206
+ console.log();
207
+ }
208
+ }
209
+ catch (error) {
210
+ spinner?.stop();
211
+ if (format === 'json') {
212
+ console.log(JSON.stringify({ error: String(error) }));
213
+ }
214
+ else {
215
+ console.log(chalk.red(`\n❌ Error: ${error}`));
216
+ }
217
+ }
218
+ }
219
+ /**
220
+ * Traits subcommand
221
+ */
222
+ async function traitsAction(targetPath, options) {
223
+ const rootDir = targetPath ?? process.cwd();
224
+ const format = options.format ?? 'text';
225
+ const isTextFormat = format === 'text';
226
+ const spinner = isTextFormat ? createSpinner('Analyzing traits...') : null;
227
+ spinner?.start();
228
+ try {
229
+ const analyzer = createRustAnalyzer({ rootDir, verbose: options.verbose ?? false });
230
+ const result = await analyzer.analyzeTraits();
231
+ spinner?.stop();
232
+ // JSON output
233
+ if (format === 'json') {
234
+ console.log(JSON.stringify(result, null, 2));
235
+ return;
236
+ }
237
+ // Text output
238
+ console.log();
239
+ console.log(chalk.bold('🔌 Rust Traits'));
240
+ console.log(chalk.gray('─'.repeat(60)));
241
+ console.log();
242
+ console.log(`Total Traits: ${chalk.cyan(result.traits.length)}`);
243
+ console.log(`Total Implementations: ${chalk.cyan(result.implementations.length)}`);
244
+ console.log();
245
+ if (result.traits.length === 0) {
246
+ console.log(chalk.gray('No traits found'));
247
+ console.log();
248
+ return;
249
+ }
250
+ for (const trait of result.traits) {
251
+ const implCount = trait.implementations.length;
252
+ const implBadge = implCount > 0 ? chalk.green(`(${implCount} impl)`) : chalk.gray('(no impl)');
253
+ console.log(`${chalk.bold(trait.name)} ${implBadge}`);
254
+ console.log(chalk.gray(` Methods: ${trait.methods.join(', ') || 'none'}`));
255
+ if (trait.implementations.length > 0) {
256
+ console.log(chalk.gray(` Implementations: ${trait.implementations.join(', ')}`));
257
+ }
258
+ console.log();
259
+ }
260
+ }
261
+ catch (error) {
262
+ spinner?.stop();
263
+ if (format === 'json') {
264
+ console.log(JSON.stringify({ error: String(error) }));
265
+ }
266
+ else {
267
+ console.log(chalk.red(`\n❌ Error: ${error}`));
268
+ }
269
+ }
270
+ }
271
+ /**
272
+ * Data access subcommand
273
+ */
274
+ async function dataAccessAction(targetPath, options) {
275
+ const rootDir = targetPath ?? process.cwd();
276
+ const format = options.format ?? 'text';
277
+ const isTextFormat = format === 'text';
278
+ const spinner = isTextFormat ? createSpinner('Analyzing data access patterns...') : null;
279
+ spinner?.start();
280
+ try {
281
+ const analyzer = createRustAnalyzer({ rootDir, verbose: options.verbose ?? false });
282
+ const result = await analyzer.analyzeDataAccess();
283
+ spinner?.stop();
284
+ // JSON output
285
+ if (format === 'json') {
286
+ console.log(JSON.stringify(result, null, 2));
287
+ return;
288
+ }
289
+ // Text output
290
+ console.log();
291
+ console.log(chalk.bold('🗄️ Rust Data Access Patterns'));
292
+ console.log(chalk.gray('─'.repeat(60)));
293
+ console.log();
294
+ console.log(`Total Access Points: ${chalk.cyan(result.accessPoints.length)}`);
295
+ console.log(`Tables: ${chalk.cyan(result.tables.length)}`);
296
+ console.log();
297
+ // By framework
298
+ console.log(chalk.bold('By Framework:'));
299
+ for (const [framework, count] of Object.entries(result.byFramework)) {
300
+ console.log(` ${framework}: ${chalk.cyan(count)}`);
301
+ }
302
+ console.log();
303
+ // By operation
304
+ console.log(chalk.bold('By Operation:'));
305
+ for (const [operation, count] of Object.entries(result.byOperation)) {
306
+ const opColor = operation === 'read' ? chalk.green :
307
+ operation === 'write' ? chalk.blue :
308
+ operation === 'delete' ? chalk.red : chalk.gray;
309
+ console.log(` ${opColor(operation)}: ${chalk.cyan(count)}`);
310
+ }
311
+ console.log();
312
+ // Tables
313
+ if (result.tables.length > 0) {
314
+ console.log(chalk.bold('Tables Accessed:'));
315
+ for (const table of result.tables) {
316
+ console.log(` • ${table}`);
317
+ }
318
+ console.log();
319
+ }
320
+ // Access points (verbose)
321
+ if (options.verbose && result.accessPoints.length > 0) {
322
+ console.log(chalk.bold('Access Points:'));
323
+ for (const ap of result.accessPoints.slice(0, 20)) {
324
+ const opColor = ap.operation === 'read' ? chalk.green :
325
+ ap.operation === 'write' ? chalk.blue :
326
+ ap.operation === 'delete' ? chalk.red : chalk.gray;
327
+ console.log(` ${opColor(ap.operation.padEnd(6))} ${ap.table} (${ap.framework})`);
328
+ console.log(chalk.gray(` ${ap.file}:${ap.line}`));
329
+ }
330
+ if (result.accessPoints.length > 20) {
331
+ console.log(chalk.gray(` ... and ${result.accessPoints.length - 20} more`));
332
+ }
333
+ console.log();
334
+ }
335
+ }
336
+ catch (error) {
337
+ spinner?.stop();
338
+ if (format === 'json') {
339
+ console.log(JSON.stringify({ error: String(error) }));
340
+ }
341
+ else {
342
+ console.log(chalk.red(`\n❌ Error: ${error}`));
343
+ }
344
+ }
345
+ }
346
+ /**
347
+ * Async subcommand
348
+ */
349
+ async function asyncAction(targetPath, options) {
350
+ const rootDir = targetPath ?? process.cwd();
351
+ const format = options.format ?? 'text';
352
+ const isTextFormat = format === 'text';
353
+ const spinner = isTextFormat ? createSpinner('Analyzing async patterns...') : null;
354
+ spinner?.start();
355
+ try {
356
+ const analyzer = createRustAnalyzer({ rootDir, verbose: options.verbose ?? false });
357
+ const result = await analyzer.analyzeAsync();
358
+ spinner?.stop();
359
+ // JSON output
360
+ if (format === 'json') {
361
+ console.log(JSON.stringify(result, null, 2));
362
+ return;
363
+ }
364
+ // Text output
365
+ console.log();
366
+ console.log(chalk.bold('⚡ Rust Async Analysis'));
367
+ console.log(chalk.gray('─'.repeat(60)));
368
+ console.log();
369
+ console.log(`Async Functions: ${chalk.cyan(result.stats.asyncFunctions)}`);
370
+ console.log(`Await Points: ${chalk.cyan(result.stats.awaitPoints)}`);
371
+ console.log(`Spawned Tasks: ${chalk.cyan(result.stats.spawnedTasks)}`);
372
+ console.log(`Channels: ${chalk.cyan(result.stats.channels)}`);
373
+ console.log(`Mutexes: ${chalk.cyan(result.stats.mutexes)}`);
374
+ console.log();
375
+ // Runtime detection
376
+ if (result.runtime) {
377
+ console.log(chalk.bold('Detected Runtime:'));
378
+ console.log(` ${chalk.cyan(result.runtime)}`);
379
+ console.log();
380
+ }
381
+ // Issues
382
+ if (result.issues.length > 0) {
383
+ console.log(chalk.bold('Potential Issues:'));
384
+ for (const issue of result.issues.slice(0, 10)) {
385
+ console.log(` ${chalk.yellow('⚠')} ${issue.message}`);
386
+ console.log(chalk.gray(` ${issue.file}:${issue.line}`));
387
+ }
388
+ if (result.issues.length > 10) {
389
+ console.log(chalk.gray(` ... and ${result.issues.length - 10} more`));
390
+ }
391
+ console.log();
392
+ }
393
+ }
394
+ catch (error) {
395
+ spinner?.stop();
396
+ if (format === 'json') {
397
+ console.log(JSON.stringify({ error: String(error) }));
398
+ }
399
+ else {
400
+ console.log(chalk.red(`\n❌ Error: ${error}`));
401
+ }
402
+ }
403
+ }
404
+ /**
405
+ * Status subcommand
406
+ */
407
+ async function statusAction(targetPath, options) {
408
+ const rootDir = targetPath ?? process.cwd();
409
+ const format = options.format ?? 'text';
410
+ const isTextFormat = format === 'text';
411
+ const spinner = isTextFormat ? createSpinner('Analyzing Rust project...') : null;
412
+ spinner?.start();
413
+ try {
414
+ const analyzer = createRustAnalyzer({ rootDir, verbose: options.verbose ?? false });
415
+ const result = await analyzer.analyze();
416
+ spinner?.stop();
417
+ // JSON output
418
+ if (format === 'json') {
419
+ console.log(JSON.stringify({
420
+ project: {
421
+ crateName: result.crateName,
422
+ edition: result.edition,
423
+ crates: result.crates.length,
424
+ files: result.stats.fileCount,
425
+ functions: result.stats.functionCount,
426
+ structs: result.stats.structCount,
427
+ traits: result.stats.traitCount,
428
+ },
429
+ frameworks: result.detectedFrameworks,
430
+ stats: result.stats,
431
+ topCrates: result.crates.slice(0, 10).map((crate) => ({
432
+ name: crate.name,
433
+ files: crate.files.length,
434
+ functions: crate.functions.length,
435
+ })),
436
+ }, null, 2));
437
+ return;
438
+ }
439
+ // Text output
440
+ console.log();
441
+ console.log(chalk.bold('📊 Rust Project Status'));
442
+ console.log(chalk.gray('═'.repeat(60)));
443
+ console.log();
444
+ // Project info
445
+ console.log(chalk.bold('Project'));
446
+ console.log(chalk.gray('─'.repeat(40)));
447
+ if (result.crateName) {
448
+ console.log(` Crate: ${chalk.cyan(result.crateName)}`);
449
+ }
450
+ if (result.edition) {
451
+ console.log(` Edition: ${chalk.cyan(result.edition)}`);
452
+ }
453
+ console.log(` Crates: ${chalk.cyan(result.crates.length)}`);
454
+ console.log(` Files: ${chalk.cyan(result.stats.fileCount)}`);
455
+ console.log();
456
+ // Detected frameworks
457
+ if (result.detectedFrameworks.length > 0) {
458
+ console.log(chalk.bold('Detected Frameworks'));
459
+ console.log(chalk.gray('─'.repeat(40)));
460
+ for (const fw of result.detectedFrameworks) {
461
+ console.log(` • ${fw}`);
462
+ }
463
+ console.log();
464
+ }
465
+ // Statistics
466
+ console.log(chalk.bold('Statistics'));
467
+ console.log(chalk.gray('─'.repeat(40)));
468
+ console.log(` Functions: ${chalk.cyan(result.stats.functionCount)}`);
469
+ console.log(` Structs: ${chalk.cyan(result.stats.structCount)}`);
470
+ console.log(` Traits: ${chalk.cyan(result.stats.traitCount)}`);
471
+ console.log(` Enums: ${chalk.cyan(result.stats.enumCount)}`);
472
+ console.log(` Lines of Code: ${chalk.cyan(result.stats.linesOfCode.toLocaleString())}`);
473
+ console.log(` Test Files: ${chalk.cyan(result.stats.testFileCount)}`);
474
+ console.log(` Test Functions: ${chalk.cyan(result.stats.testFunctionCount)}`);
475
+ console.log(` Analysis Time: ${chalk.gray(`${result.stats.analysisTimeMs.toFixed(0)}ms`)}`);
476
+ console.log();
477
+ // Top crates
478
+ if (result.crates.length > 0) {
479
+ console.log(chalk.bold('Top Crates'));
480
+ console.log(chalk.gray('─'.repeat(40)));
481
+ for (const crate of result.crates.slice(0, 5)) {
482
+ console.log(` ${chalk.white(crate.name)}`);
483
+ console.log(chalk.gray(` ${crate.files.length} files, ${crate.functions.length} functions`));
484
+ }
485
+ if (result.crates.length > 5) {
486
+ console.log(chalk.gray(` ... and ${result.crates.length - 5} more crates`));
487
+ }
488
+ console.log();
489
+ }
490
+ // Next steps
491
+ console.log(chalk.gray('─'.repeat(60)));
492
+ console.log(chalk.bold('📌 Next Steps:'));
493
+ console.log(chalk.gray(` • drift rust routes ${chalk.white('View HTTP routes')}`));
494
+ console.log(chalk.gray(` • drift rust errors ${chalk.white('Analyze error handling')}`));
495
+ console.log(chalk.gray(` • drift rust traits ${chalk.white('View traits')}`));
496
+ console.log(chalk.gray(` • drift rust data-access ${chalk.white('View data access patterns')}`));
497
+ console.log(chalk.gray(` • drift rust async ${chalk.white('Analyze async patterns')}`));
498
+ console.log();
499
+ }
500
+ catch (error) {
501
+ spinner?.stop();
502
+ if (format === 'json') {
503
+ console.log(JSON.stringify({ error: String(error) }));
504
+ }
505
+ else {
506
+ console.log(chalk.red(`\n❌ Error: ${error}`));
507
+ }
508
+ }
509
+ }
510
+ /**
511
+ * Get color for HTTP method
512
+ */
513
+ function getMethodColor(method) {
514
+ const colors = {
515
+ GET: chalk.green,
516
+ POST: chalk.blue,
517
+ PUT: chalk.yellow,
518
+ DELETE: chalk.red,
519
+ PATCH: chalk.magenta,
520
+ HEAD: chalk.cyan,
521
+ OPTIONS: chalk.gray,
522
+ ANY: chalk.white,
523
+ };
524
+ return colors[method] ?? chalk.white;
525
+ }
526
+ //# sourceMappingURL=rust.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rust.js","sourceRoot":"","sources":["../../src/commands/rust.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"scan.d.ts","sourceRoot":"","sources":["../../src/commands/scan.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAyBpC,MAAM,WAAW,kBAAkB;IACjC,6BAA6B;IAC7B,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;IACjB,4BAA4B;IAC5B,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,0CAA0C;IAC1C,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,kCAAkC;IAClC,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,2BAA2B;IAC3B,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,gDAAgD;IAChD,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,4CAA4C;IAC5C,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,kEAAkE;IAClE,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,kEAAkE;IAClE,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,iDAAiD;IACjD,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,mCAAmC;IACnC,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,yDAAyD;IACzD,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAggCD,eAAO,MAAM,WAAW,SAwBpB,CAAC"}
1
+ {"version":3,"file":"scan.d.ts","sourceRoot":"","sources":["../../src/commands/scan.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AA2BpC,MAAM,WAAW,kBAAkB;IACjC,6BAA6B;IAC7B,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;IACjB,4BAA4B;IAC5B,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,0CAA0C;IAC1C,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,kCAAkC;IAClC,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,2BAA2B;IAC3B,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,gDAAgD;IAChD,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,4CAA4C;IAC5C,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,kEAAkE;IAClE,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,kEAAkE;IAClE,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,iDAAiD;IACjD,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,mCAAmC;IACnC,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,yDAAyD;IACzD,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AA2kCD,eAAO,MAAM,WAAW,SAwBpB,CAAC"}
@@ -11,7 +11,7 @@ import * as fs from 'node:fs/promises';
11
11
  import * as path from 'node:path';
12
12
  import * as crypto from 'node:crypto';
13
13
  import chalk from 'chalk';
14
- import { PatternStore, HistoryStore, FileWalker, createDataLake, loadProjectConfig, getProjectRegistry, } from 'driftdetect-core';
14
+ import { PatternStore, HistoryStore, FileWalker, createDataLake, loadProjectConfig, getProjectRegistry, createTelemetryClient, } from 'driftdetect-core';
15
15
  import { createSpinner, status } from '../ui/spinner.js';
16
16
  import { createPatternsTable } from '../ui/table.js';
17
17
  import { createScannerService } from '../services/scanner-service.js';
@@ -690,6 +690,74 @@ async function scanSingleProject(rootDir, options, quiet = false) {
690
690
  console.log(chalk.cyan(' drift where <pattern>'));
691
691
  console.log(chalk.cyan(' drift files <path>'));
692
692
  }
693
+ // Record telemetry (if enabled) - inside try block where scanResults is in scope
694
+ try {
695
+ const projectConfig = await loadProjectConfig(rootDir);
696
+ if (projectConfig.telemetry?.enabled) {
697
+ const driftDir = path.join(rootDir, DRIFT_DIR);
698
+ const telemetryClient = createTelemetryClient(driftDir, projectConfig.telemetry);
699
+ await telemetryClient.initialize();
700
+ // Record scan completion
701
+ const scanDuration = Date.now() - startTime;
702
+ await telemetryClient.recordScanCompletion({
703
+ durationMs: scanDuration,
704
+ filesScanned: files.length,
705
+ newPatternsDiscovered: addedCount,
706
+ isIncremental: options.incremental ?? false,
707
+ workerCount: scannerService.getWorkerThreadCount(),
708
+ });
709
+ // Record pattern signatures for discovered patterns (limit to 50)
710
+ for (const aggPattern of scanResults.patterns.slice(0, 50)) {
711
+ await telemetryClient.recordPatternSignature({
712
+ patternName: aggPattern.name,
713
+ detectorConfig: { detectorId: aggPattern.detectorId },
714
+ category: aggPattern.category,
715
+ confidence: aggPattern.confidence,
716
+ locationCount: aggPattern.locations.length,
717
+ outlierCount: 0,
718
+ detectionMethod: 'regex',
719
+ language: getExtension(aggPattern.locations[0]?.file ?? 'ts'),
720
+ });
721
+ }
722
+ // Record aggregate stats
723
+ const telemetryPatterns = store.getAll();
724
+ const patternsByCategory = {};
725
+ for (const p of telemetryPatterns) {
726
+ patternsByCategory[p.category] = (patternsByCategory[p.category] ?? 0) + 1;
727
+ }
728
+ const languages = new Set();
729
+ for (const file of files) {
730
+ languages.add(getExtension(file));
731
+ }
732
+ await telemetryClient.recordAggregateStats({
733
+ totalPatterns: telemetryPatterns.length,
734
+ patternsByStatus: {
735
+ discovered: store.getDiscovered().length,
736
+ approved: store.getApproved().length,
737
+ ignored: store.getIgnored().length,
738
+ },
739
+ patternsByCategory,
740
+ languages: Array.from(languages),
741
+ frameworks: [],
742
+ featuresEnabled: [
743
+ options.contracts !== false ? 'contracts' : '',
744
+ options.boundaries !== false ? 'boundaries' : '',
745
+ options.manifest ? 'manifest' : '',
746
+ ].filter(Boolean),
747
+ fileCount: files.length,
748
+ });
749
+ await telemetryClient.shutdown();
750
+ if (verbose) {
751
+ console.log(chalk.gray(' Telemetry events submitted'));
752
+ }
753
+ }
754
+ }
755
+ catch (telemetryError) {
756
+ // Telemetry should never block - silently ignore errors
757
+ if (verbose) {
758
+ console.log(chalk.gray(` Telemetry: ${telemetryError.message}`));
759
+ }
760
+ }
693
761
  }
694
762
  catch (error) {
695
763
  scanSpinner.fail('Scan failed');