sublyzer-snapshot 0.3.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 (115) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +306 -0
  3. package/dist/api/sublyzer.d.ts +29 -0
  4. package/dist/api/sublyzer.d.ts.map +1 -0
  5. package/dist/api/sublyzer.js +61 -0
  6. package/dist/api/sublyzer.js.map +1 -0
  7. package/dist/cli.d.ts +3 -0
  8. package/dist/cli.d.ts.map +1 -0
  9. package/dist/cli.js +199 -0
  10. package/dist/cli.js.map +1 -0
  11. package/dist/commands/ci.d.ts +9 -0
  12. package/dist/commands/ci.d.ts.map +1 -0
  13. package/dist/commands/ci.js +65 -0
  14. package/dist/commands/ci.js.map +1 -0
  15. package/dist/commands/compare.d.ts +7 -0
  16. package/dist/commands/compare.d.ts.map +1 -0
  17. package/dist/commands/compare.js +60 -0
  18. package/dist/commands/compare.js.map +1 -0
  19. package/dist/commands/doctor.d.ts +13 -0
  20. package/dist/commands/doctor.d.ts.map +1 -0
  21. package/dist/commands/doctor.js +100 -0
  22. package/dist/commands/doctor.js.map +1 -0
  23. package/dist/commands/init.d.ts +10 -0
  24. package/dist/commands/init.d.ts.map +1 -0
  25. package/dist/commands/init.js +88 -0
  26. package/dist/commands/init.js.map +1 -0
  27. package/dist/commands/open.d.ts +2 -0
  28. package/dist/commands/open.d.ts.map +1 -0
  29. package/dist/commands/open.js +16 -0
  30. package/dist/commands/open.js.map +1 -0
  31. package/dist/commands/pull.d.ts +9 -0
  32. package/dist/commands/pull.d.ts.map +1 -0
  33. package/dist/commands/pull.js +34 -0
  34. package/dist/commands/pull.js.map +1 -0
  35. package/dist/commands/report.d.ts +8 -0
  36. package/dist/commands/report.d.ts.map +1 -0
  37. package/dist/commands/report.js +47 -0
  38. package/dist/commands/report.js.map +1 -0
  39. package/dist/commands/run.d.ts +21 -0
  40. package/dist/commands/run.d.ts.map +1 -0
  41. package/dist/commands/run.js +96 -0
  42. package/dist/commands/run.js.map +1 -0
  43. package/dist/commands/status.d.ts +5 -0
  44. package/dist/commands/status.d.ts.map +1 -0
  45. package/dist/commands/status.js +52 -0
  46. package/dist/commands/status.js.map +1 -0
  47. package/dist/config.d.ts +36 -0
  48. package/dist/config.d.ts.map +1 -0
  49. package/dist/config.js +58 -0
  50. package/dist/config.js.map +1 -0
  51. package/dist/constants.d.ts +13 -0
  52. package/dist/constants.d.ts.map +1 -0
  53. package/dist/constants.js +12 -0
  54. package/dist/constants.js.map +1 -0
  55. package/dist/detect/git.d.ts +9 -0
  56. package/dist/detect/git.d.ts.map +1 -0
  57. package/dist/detect/git.js +25 -0
  58. package/dist/detect/git.js.map +1 -0
  59. package/dist/detect/meta.d.ts +8 -0
  60. package/dist/detect/meta.d.ts.map +1 -0
  61. package/dist/detect/meta.js +42 -0
  62. package/dist/detect/meta.js.map +1 -0
  63. package/dist/detect/routes.d.ts +2 -0
  64. package/dist/detect/routes.d.ts.map +1 -0
  65. package/dist/detect/routes.js +104 -0
  66. package/dist/detect/routes.js.map +1 -0
  67. package/dist/detect/stack.d.ts +15 -0
  68. package/dist/detect/stack.d.ts.map +1 -0
  69. package/dist/detect/stack.js +114 -0
  70. package/dist/detect/stack.js.map +1 -0
  71. package/dist/detect/workspaces.d.ts +6 -0
  72. package/dist/detect/workspaces.d.ts.map +1 -0
  73. package/dist/detect/workspaces.js +54 -0
  74. package/dist/detect/workspaces.js.map +1 -0
  75. package/dist/report/markdown.d.ts +5 -0
  76. package/dist/report/markdown.d.ts.map +1 -0
  77. package/dist/report/markdown.js +81 -0
  78. package/dist/report/markdown.js.map +1 -0
  79. package/dist/scan/audit.d.ts +16 -0
  80. package/dist/scan/audit.d.ts.map +1 -0
  81. package/dist/scan/audit.js +53 -0
  82. package/dist/scan/audit.js.map +1 -0
  83. package/dist/scan/health-score.d.ts +13 -0
  84. package/dist/scan/health-score.d.ts.map +1 -0
  85. package/dist/scan/health-score.js +60 -0
  86. package/dist/scan/health-score.js.map +1 -0
  87. package/dist/scan/history.d.ts +21 -0
  88. package/dist/scan/history.d.ts.map +1 -0
  89. package/dist/scan/history.js +82 -0
  90. package/dist/scan/history.js.map +1 -0
  91. package/dist/scan/outdated.d.ts +15 -0
  92. package/dist/scan/outdated.d.ts.map +1 -0
  93. package/dist/scan/outdated.js +54 -0
  94. package/dist/scan/outdated.js.map +1 -0
  95. package/dist/scan/policy.d.ts +5 -0
  96. package/dist/scan/policy.d.ts.map +1 -0
  97. package/dist/scan/policy.js +20 -0
  98. package/dist/scan/policy.js.map +1 -0
  99. package/dist/scan/snapshot.d.ts +47 -0
  100. package/dist/scan/snapshot.d.ts.map +1 -0
  101. package/dist/scan/snapshot.js +176 -0
  102. package/dist/scan/snapshot.js.map +1 -0
  103. package/dist/utils/gitignore.d.ts +3 -0
  104. package/dist/utils/gitignore.d.ts.map +1 -0
  105. package/dist/utils/gitignore.js +34 -0
  106. package/dist/utils/gitignore.js.map +1 -0
  107. package/dist/utils/log.d.ts +6 -0
  108. package/dist/utils/log.d.ts.map +1 -0
  109. package/dist/utils/log.js +17 -0
  110. package/dist/utils/log.js.map +1 -0
  111. package/dist/utils/prompt.d.ts +4 -0
  112. package/dist/utils/prompt.d.ts.map +1 -0
  113. package/dist/utils/prompt.js +35 -0
  114. package/dist/utils/prompt.js.map +1 -0
  115. package/package.json +51 -0
@@ -0,0 +1,96 @@
1
+ import { dashboardIntegrationUrl, loadConfig, saveConfig } from '../config.js';
2
+ import { pushSnapshot } from '../api/sublyzer.js';
3
+ import { diffSnapshots, loadLastSnapshot, saveScanHistory, } from '../scan/history.js';
4
+ import { failOnMessage, shouldFailOnVulns } from '../scan/policy.js';
5
+ import { buildProjectSnapshot, printLocalSummary, snapshotToCollectItems, } from '../scan/snapshot.js';
6
+ import { info, ok, title, warn } from '../utils/log.js';
7
+ function applyScanToConfig(config, snapshot, eventsSent) {
8
+ config.updatedAt = new Date().toISOString();
9
+ config.lastScanAt = snapshot.scannedAt;
10
+ const lastScan = {
11
+ scannedAt: snapshot.scannedAt,
12
+ routeCount: snapshot.summary.routeCount,
13
+ dependencyCount: snapshot.dependencyCount,
14
+ vulnerablePackages: snapshot.summary.vulnerablePackages,
15
+ criticalVulns: snapshot.summary.criticalVulns,
16
+ highVulns: snapshot.summary.highVulns,
17
+ eventsSent,
18
+ healthScore: snapshot.health.score,
19
+ healthGrade: snapshot.health.grade,
20
+ };
21
+ config.lastScan = lastScan;
22
+ }
23
+ export async function runScan(opts = {}) {
24
+ if (!opts.json)
25
+ title('Sublyzer Snapshot — scan');
26
+ const config = loadConfig();
27
+ const root = config.projectRoot || process.cwd();
28
+ const previous = loadLastSnapshot(root);
29
+ if (!opts.json)
30
+ info('Scanning project…');
31
+ const snapshot = buildProjectSnapshot(root, {
32
+ skipAudit: opts.skipAudit,
33
+ skipOutdated: opts.skipOutdated,
34
+ });
35
+ const healthDelta = previous
36
+ ? snapshot.health.score - (previous.health?.score ?? previous.summary?.healthScore ?? 0)
37
+ : null;
38
+ const diff = diffSnapshots(previous, snapshot, healthDelta);
39
+ if (!opts.json && previous && (diff.routesAdded.length || diff.vulnDelta.total !== 0 || healthDelta !== 0)) {
40
+ info('Changes since last scan:');
41
+ if (healthDelta != null && healthDelta !== 0) {
42
+ console.log(` Health: ${healthDelta > 0 ? '+' : ''}${healthDelta} → ${snapshot.health.score}/100`);
43
+ }
44
+ if (diff.vulnDelta.total !== 0) {
45
+ console.log(` Vulnerabilities: ${diff.vulnDelta.total >= 0 ? '+' : ''}${diff.vulnDelta.total}`);
46
+ }
47
+ if (diff.routesAdded.length)
48
+ console.log(` Routes added: ${diff.routesAdded.length}`);
49
+ if (diff.routesRemoved.length)
50
+ console.log(` Routes removed: ${diff.routesRemoved.length}`);
51
+ console.log('');
52
+ }
53
+ if (!opts.json)
54
+ printLocalSummary(snapshot);
55
+ if (opts.failOn && shouldFailOnVulns(snapshot, opts.failOn)) {
56
+ const msg = failOnMessage(snapshot, opts.failOn);
57
+ if (opts.json) {
58
+ return { success: false, dryRun: Boolean(opts.dryRun), snapshot, diff, policyFailed: true };
59
+ }
60
+ throw new Error(msg);
61
+ }
62
+ if (opts.dryRun) {
63
+ if (!opts.json)
64
+ warn('Dry run — nothing sent to Sublyzer.');
65
+ return { success: true, dryRun: true, snapshot, diff };
66
+ }
67
+ const items = snapshotToCollectItems(snapshot);
68
+ if (!opts.json)
69
+ info(`Pushing ${items.length} events to ${config.apiUrl}…`);
70
+ const result = await pushSnapshot(config.apiUrl, config.integrationCode, items);
71
+ if (!result.success) {
72
+ throw new Error(result.error || 'Failed to push snapshot');
73
+ }
74
+ const eventsSent = result.processed ?? items.length;
75
+ const dash = dashboardIntegrationUrl(config);
76
+ saveScanHistory(snapshot, root);
77
+ applyScanToConfig(config, snapshot, eventsSent);
78
+ saveConfig(config, root);
79
+ const runResult = {
80
+ success: true,
81
+ dryRun: false,
82
+ snapshot,
83
+ eventsSent,
84
+ dashboardUrl: dash,
85
+ diff,
86
+ };
87
+ if (opts.json)
88
+ return runResult;
89
+ ok(`Sent ${eventsSent} events`);
90
+ console.log(` Dashboard: ${dash}`);
91
+ console.log('');
92
+ console.log(' Next: sublyzer-snapshot report | compare | pull');
93
+ console.log('');
94
+ return runResult;
95
+ }
96
+ //# sourceMappingURL=run.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run.js","sourceRoot":"","sources":["../../src/commands/run.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export type StatusOptions = {
2
+ json?: boolean;
3
+ };
4
+ export declare function runStatus(opts?: StatusOptions): Promise<Record<string, unknown>>;
5
+ //# sourceMappingURL=status.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"status.d.ts","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"AAUA,MAAM,MAAM,aAAa,GAAG;IAAE,IAAI,CAAC,EAAE,OAAO,CAAA;CAAE,CAAC;AAE/C,wBAAsB,SAAS,CAAC,IAAI,GAAE,aAAkB,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAqD1F"}
@@ -0,0 +1,52 @@
1
+ import { configExists, configPath, dashboardIntegrationUrl, loadConfig, maskSecret, publicReadUrl, } from '../config.js';
2
+ import { info, title } from '../utils/log.js';
3
+ export async function runStatus(opts = {}) {
4
+ if (!configExists()) {
5
+ throw new Error(`Not initialized. Run: sublyzer-snapshot init`);
6
+ }
7
+ const config = loadConfig();
8
+ const payload = {
9
+ initialized: true,
10
+ configPath: configPath(),
11
+ projectName: config.projectName,
12
+ projectRoot: config.projectRoot,
13
+ stack: config.stack,
14
+ integration: {
15
+ id: config.integrationId,
16
+ name: config.integrationName,
17
+ code: maskSecret(config.integrationCode, 6),
18
+ status: 'linked',
19
+ },
20
+ apiUrl: config.apiUrl,
21
+ dashboardUrl: dashboardIntegrationUrl(config),
22
+ readKey: config.readKey ? maskSecret(config.readKey) : null,
23
+ pullEnabled: Boolean(config.readKey || process.env.SUBLYZER_READ_KEY),
24
+ publicReadUrl: publicReadUrl(config, { limit: 50, windowDays: 7 }),
25
+ createdAt: config.createdAt,
26
+ updatedAt: config.updatedAt,
27
+ lastScanAt: config.lastScanAt ?? null,
28
+ lastScan: config.lastScan ?? null,
29
+ };
30
+ if (opts.json) {
31
+ return payload;
32
+ }
33
+ title('Sublyzer Snapshot — status');
34
+ info(`Project: ${config.projectName}`);
35
+ info(`Config: ${configPath()}`);
36
+ info(`Stack: ${config.stack}`);
37
+ info(`Integration: ${config.integrationName || '—'} (${maskSecret(config.integrationCode, 6)})`);
38
+ info(`Dashboard: ${dashboardIntegrationUrl(config)}`);
39
+ info(`Read key: ${config.readKey ? maskSecret(config.readKey) : '(not set — add via init or SUBLYZER_READ_KEY)'}`);
40
+ if (config.lastScanAt) {
41
+ info(`Last scan: ${config.lastScanAt}`);
42
+ if (config.lastScan) {
43
+ info(` health ${config.lastScan.healthScore}/100 (${config.lastScan.healthGrade}), routes ${config.lastScan.routeCount}, vulns ${config.lastScan.vulnerablePackages}`);
44
+ }
45
+ }
46
+ else {
47
+ info('Last scan: never — run `sublyzer-snapshot run`');
48
+ }
49
+ console.log('');
50
+ return payload;
51
+ }
52
+ //# sourceMappingURL=status.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"status.js","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,YAAY,EACZ,UAAU,EACV,uBAAuB,EACvB,UAAU,EACV,UAAU,EACV,aAAa,GACd,MAAM,cAAc,CAAC;AACtB,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAI9C,MAAM,CAAC,KAAK,UAAU,SAAS,CAAC,OAAsB,EAAE;IACtD,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;QACpB,MAAM,IAAI,KAAK,CAAC,8CAA8C,CAAC,CAAC;IAClE,CAAC;IAED,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,OAAO,GAAG;QACd,WAAW,EAAE,IAAI;QACjB,UAAU,EAAE,UAAU,EAAE;QACxB,WAAW,EAAE,MAAM,CAAC,WAAW;QAC/B,WAAW,EAAE,MAAM,CAAC,WAAW;QAC/B,KAAK,EAAE,MAAM,CAAC,KAAK;QACnB,WAAW,EAAE;YACX,EAAE,EAAE,MAAM,CAAC,aAAa;YACxB,IAAI,EAAE,MAAM,CAAC,eAAe;YAC5B,IAAI,EAAE,UAAU,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC,CAAC;YAC3C,MAAM,EAAE,QAAQ;SACjB;QACD,MAAM,EAAE,MAAM,CAAC,MAAM;QACrB,YAAY,EAAE,uBAAuB,CAAC,MAAM,CAAC;QAC7C,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI;QAC3D,WAAW,EAAE,OAAO,CAAC,MAAM,CAAC,OAAO,IAAI,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC;QACrE,aAAa,EAAE,aAAa,CAAC,MAAM,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,UAAU,EAAE,CAAC,EAAE,CAAC;QAClE,SAAS,EAAE,MAAM,CAAC,SAAS;QAC3B,SAAS,EAAE,MAAM,CAAC,SAAS;QAC3B,UAAU,EAAE,MAAM,CAAC,UAAU,IAAI,IAAI;QACrC,QAAQ,EAAE,MAAM,CAAC,QAAQ,IAAI,IAAI;KAClC,CAAC;IAEF,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,4BAA4B,CAAC,CAAC;IACpC,IAAI,CAAC,gBAAgB,MAAM,CAAC,WAAW,EAAE,CAAC,CAAC;IAC3C,IAAI,CAAC,gBAAgB,UAAU,EAAE,EAAE,CAAC,CAAC;IACrC,IAAI,CAAC,gBAAgB,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;IACrC,IAAI,CAAC,gBAAgB,MAAM,CAAC,eAAe,IAAI,GAAG,KAAK,UAAU,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC;IACjG,IAAI,CAAC,gBAAgB,uBAAuB,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IACxD,IAAI,CAAC,gBAAgB,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,UAAU,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,+CAA+C,EAAE,CAAC,CAAC;IACtH,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;QACtB,IAAI,CAAC,gBAAgB,MAAM,CAAC,UAAU,EAAE,CAAC,CAAC;QAC1C,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;YACpB,IAAI,CACF,YAAY,MAAM,CAAC,QAAQ,CAAC,WAAW,SAAS,MAAM,CAAC,QAAQ,CAAC,WAAW,aAAa,MAAM,CAAC,QAAQ,CAAC,UAAU,WAAW,MAAM,CAAC,QAAQ,CAAC,kBAAkB,EAAE,CAClK,CAAC;QACJ,CAAC;IACH,CAAC;SAAM,CAAC;QACN,IAAI,CAAC,kDAAkD,CAAC,CAAC;IAC3D,CAAC;IACD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAEhB,OAAO,OAAO,CAAC;AACjB,CAAC"}
@@ -0,0 +1,36 @@
1
+ export type LastScanSummary = {
2
+ scannedAt: string;
3
+ routeCount: number;
4
+ dependencyCount: number;
5
+ vulnerablePackages: number;
6
+ criticalVulns: number;
7
+ highVulns: number;
8
+ eventsSent: number;
9
+ healthScore: number;
10
+ healthGrade: 'A' | 'B' | 'C' | 'D' | 'F';
11
+ };
12
+ export type SnapshotConfig = {
13
+ version: 1;
14
+ integrationCode: string;
15
+ apiUrl: string;
16
+ dashboardUrl: string;
17
+ integrationId?: string;
18
+ integrationName?: string;
19
+ readKey?: string;
20
+ projectName: string;
21
+ projectRoot: string;
22
+ stack: string;
23
+ createdAt: string;
24
+ updatedAt: string;
25
+ lastScanAt?: string;
26
+ lastScan?: LastScanSummary;
27
+ };
28
+ export declare function configPath(cwd?: string): string;
29
+ export declare function configExists(cwd?: string): boolean;
30
+ export declare function loadConfig(cwd?: string): SnapshotConfig;
31
+ export declare function saveConfig(config: SnapshotConfig, cwd?: string): void;
32
+ export declare function dashboardIntegrationUrl(config: SnapshotConfig): string;
33
+ export declare function publicReadUrl(config: SnapshotConfig, params?: Record<string, string | number>): string | null;
34
+ export declare function maskSecret(value: string | undefined, visible?: number): string;
35
+ export declare function resolveReadKey(config: SnapshotConfig, override?: string): string | undefined;
36
+ //# sourceMappingURL=config.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAIA,MAAM,MAAM,eAAe,GAAG;IAC5B,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,eAAe,EAAE,MAAM,CAAC;IACxB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,aAAa,EAAE,MAAM,CAAC;IACtB,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;CAC1C,CAAC;AAEF,MAAM,MAAM,cAAc,GAAG;IAC3B,OAAO,EAAE,CAAC,CAAC;IACX,eAAe,EAAE,MAAM,CAAC;IACxB,MAAM,EAAE,MAAM,CAAC;IACf,YAAY,EAAE,MAAM,CAAC;IACrB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,EAAE,eAAe,CAAC;CAC5B,CAAC;AAEF,wBAAgB,UAAU,CAAC,GAAG,SAAgB,GAAG,MAAM,CAEtD;AAED,wBAAgB,YAAY,CAAC,GAAG,SAAgB,GAAG,OAAO,CAEzD;AAED,wBAAgB,UAAU,CAAC,GAAG,SAAgB,GAAG,cAAc,CAU9D;AAED,wBAAgB,UAAU,CAAC,MAAM,EAAE,cAAc,EAAE,GAAG,SAAgB,GAAG,IAAI,CAI5E;AAED,wBAAgB,uBAAuB,CAAC,MAAM,EAAE,cAAc,GAAG,MAAM,CAMtE;AAED,wBAAgB,aAAa,CAAC,MAAM,EAAE,cAAc,EAAE,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,CAAC,GAAG,MAAM,GAAG,IAAI,CAW7G;AAED,wBAAgB,UAAU,CAAC,KAAK,EAAE,MAAM,GAAG,SAAS,EAAE,OAAO,SAAI,GAAG,MAAM,CAIzE;AAED,wBAAgB,cAAc,CAAC,MAAM,EAAE,cAAc,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS,CAI5F"}
package/dist/config.js ADDED
@@ -0,0 +1,58 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ import { CONFIG_DIR, CONFIG_FILE } from './constants.js';
4
+ export function configPath(cwd = process.cwd()) {
5
+ return path.join(cwd, CONFIG_DIR, CONFIG_FILE);
6
+ }
7
+ export function configExists(cwd = process.cwd()) {
8
+ return fs.existsSync(configPath(cwd));
9
+ }
10
+ export function loadConfig(cwd = process.cwd()) {
11
+ const file = configPath(cwd);
12
+ if (!fs.existsSync(file)) {
13
+ throw new Error(`Config not found at ${file}. Run: sublyzer-snapshot init`);
14
+ }
15
+ const raw = JSON.parse(fs.readFileSync(file, 'utf8'));
16
+ if (raw.version !== 1) {
17
+ throw new Error('Unsupported config version. Re-run: sublyzer-snapshot init');
18
+ }
19
+ return raw;
20
+ }
21
+ export function saveConfig(config, cwd = process.cwd()) {
22
+ const file = configPath(cwd);
23
+ fs.mkdirSync(path.dirname(file), { recursive: true });
24
+ fs.writeFileSync(file, `${JSON.stringify(config, null, 2)}\n`, 'utf8');
25
+ }
26
+ export function dashboardIntegrationUrl(config) {
27
+ const base = config.dashboardUrl.replace(/\/$/, '');
28
+ if (config.integrationId) {
29
+ return `${base}/dashboard/integration/${config.integrationId}`;
30
+ }
31
+ return `${base}/dashboard`;
32
+ }
33
+ export function publicReadUrl(config, params) {
34
+ if (!config.readKey)
35
+ return null;
36
+ const base = config.apiUrl.replace(/\/$/, '');
37
+ const url = new URL(`${base}/data-collection/integration/${config.integrationCode}/data`);
38
+ url.searchParams.set('key', config.readKey);
39
+ if (params) {
40
+ for (const [k, v] of Object.entries(params)) {
41
+ url.searchParams.set(k, String(v));
42
+ }
43
+ }
44
+ return url.toString();
45
+ }
46
+ export function maskSecret(value, visible = 4) {
47
+ if (!value)
48
+ return '(not set)';
49
+ if (value.length <= visible * 2)
50
+ return '****';
51
+ return `${value.slice(0, visible)}…${value.slice(-visible)}`;
52
+ }
53
+ export function resolveReadKey(config, override) {
54
+ const fromEnv = process.env.SUBLYZER_READ_KEY?.trim();
55
+ const key = (override || fromEnv || config.readKey || '').trim();
56
+ return key || undefined;
57
+ }
58
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ export declare const SDK_NAME = "sublyzer-snapshot";
2
+ export declare const SDK_VERSION = "0.3.0";
3
+ export declare const DEFAULT_API_URL = "https://api.sublyzer.com";
4
+ export declare const DEFAULT_DASHBOARD_URL = "https://sublyzer.com";
5
+ export declare const CONFIG_DIR = ".sublyzer";
6
+ export declare const CONFIG_FILE = "snapshot.json";
7
+ export declare const LAST_SNAPSHOT_FILE = "last-snapshot.json";
8
+ export declare const HISTORY_DIR = "history";
9
+ export declare const MAX_HISTORY_FILES = 20;
10
+ export declare const INTEGRATION_CODE_RE: RegExp;
11
+ export declare const GITIGNORE_ENTRY = ".sublyzer/";
12
+ export type FailOnLevel = 'critical' | 'high' | 'moderate' | 'any';
13
+ //# sourceMappingURL=constants.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../src/constants.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,QAAQ,sBAAsB,CAAC;AAC5C,eAAO,MAAM,WAAW,UAAU,CAAC;AAEnC,eAAO,MAAM,eAAe,6BAA6B,CAAC;AAC1D,eAAO,MAAM,qBAAqB,yBAAyB,CAAC;AAE5D,eAAO,MAAM,UAAU,cAAc,CAAC;AACtC,eAAO,MAAM,WAAW,kBAAkB,CAAC;AAC3C,eAAO,MAAM,kBAAkB,uBAAuB,CAAC;AACvD,eAAO,MAAM,WAAW,YAAY,CAAC;AACrC,eAAO,MAAM,iBAAiB,KAAK,CAAC;AAEpC,eAAO,MAAM,mBAAmB,QAAmB,CAAC;AAEpD,eAAO,MAAM,eAAe,eAAe,CAAC;AAE5C,MAAM,MAAM,WAAW,GAAG,UAAU,GAAG,MAAM,GAAG,UAAU,GAAG,KAAK,CAAC"}
@@ -0,0 +1,12 @@
1
+ export const SDK_NAME = 'sublyzer-snapshot';
2
+ export const SDK_VERSION = '0.3.0';
3
+ export const DEFAULT_API_URL = 'https://api.sublyzer.com';
4
+ export const DEFAULT_DASHBOARD_URL = 'https://sublyzer.com';
5
+ export const CONFIG_DIR = '.sublyzer';
6
+ export const CONFIG_FILE = 'snapshot.json';
7
+ export const LAST_SNAPSHOT_FILE = 'last-snapshot.json';
8
+ export const HISTORY_DIR = 'history';
9
+ export const MAX_HISTORY_FILES = 20;
10
+ export const INTEGRATION_CODE_RE = /^[A-Z0-9]{24}$/;
11
+ export const GITIGNORE_ENTRY = '.sublyzer/';
12
+ //# sourceMappingURL=constants.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.js","sourceRoot":"","sources":["../src/constants.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,QAAQ,GAAG,mBAAmB,CAAC;AAC5C,MAAM,CAAC,MAAM,WAAW,GAAG,OAAO,CAAC;AAEnC,MAAM,CAAC,MAAM,eAAe,GAAG,0BAA0B,CAAC;AAC1D,MAAM,CAAC,MAAM,qBAAqB,GAAG,sBAAsB,CAAC;AAE5D,MAAM,CAAC,MAAM,UAAU,GAAG,WAAW,CAAC;AACtC,MAAM,CAAC,MAAM,WAAW,GAAG,eAAe,CAAC;AAC3C,MAAM,CAAC,MAAM,kBAAkB,GAAG,oBAAoB,CAAC;AACvD,MAAM,CAAC,MAAM,WAAW,GAAG,SAAS,CAAC;AACrC,MAAM,CAAC,MAAM,iBAAiB,GAAG,EAAE,CAAC;AAEpC,MAAM,CAAC,MAAM,mBAAmB,GAAG,gBAAgB,CAAC;AAEpD,MAAM,CAAC,MAAM,eAAe,GAAG,YAAY,CAAC"}
@@ -0,0 +1,9 @@
1
+ export type GitInfo = {
2
+ available: boolean;
3
+ branch?: string;
4
+ commit?: string;
5
+ dirty?: boolean;
6
+ remote?: string;
7
+ };
8
+ export declare function detectGit(root?: string): GitInfo;
9
+ //# sourceMappingURL=git.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git.d.ts","sourceRoot":"","sources":["../../src/detect/git.ts"],"names":[],"mappings":"AAEA,MAAM,MAAM,OAAO,GAAG;IACpB,SAAS,EAAE,OAAO,CAAC;IACnB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAQF,wBAAgB,SAAS,CAAC,IAAI,SAAgB,GAAG,OAAO,CAkBvD"}
@@ -0,0 +1,25 @@
1
+ import { spawnSync } from 'node:child_process';
2
+ function runGit(args, root) {
3
+ const res = spawnSync('git', args, { cwd: root, encoding: 'utf8', timeout: 10_000 });
4
+ if (res.status !== 0)
5
+ return null;
6
+ return (res.stdout || '').trim() || null;
7
+ }
8
+ export function detectGit(root = process.cwd()) {
9
+ const inside = runGit(['rev-parse', '--is-inside-work-tree'], root);
10
+ if (inside !== 'true') {
11
+ return { available: false };
12
+ }
13
+ const branch = runGit(['rev-parse', '--abbrev-ref', 'HEAD'], root) || undefined;
14
+ const commit = runGit(['rev-parse', '--short', 'HEAD'], root) || undefined;
15
+ const dirtyOut = runGit(['status', '--porcelain'], root);
16
+ const remote = runGit(['remote', 'get-url', 'origin'], root) || undefined;
17
+ return {
18
+ available: true,
19
+ branch,
20
+ commit,
21
+ dirty: dirtyOut ? dirtyOut.length > 0 : undefined,
22
+ remote,
23
+ };
24
+ }
25
+ //# sourceMappingURL=git.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git.js","sourceRoot":"","sources":["../../src/detect/git.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAU/C,SAAS,MAAM,CAAC,IAAc,EAAE,IAAY;IAC1C,MAAM,GAAG,GAAG,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC,CAAC;IACrF,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,IAAI,CAAC;IAClC,OAAO,CAAC,GAAG,CAAC,MAAM,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,IAAI,IAAI,CAAC;AAC3C,CAAC;AAED,MAAM,UAAU,SAAS,CAAC,IAAI,GAAG,OAAO,CAAC,GAAG,EAAE;IAC5C,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,WAAW,EAAE,uBAAuB,CAAC,EAAE,IAAI,CAAC,CAAC;IACpE,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;IAC9B,CAAC;IAED,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,WAAW,EAAE,cAAc,EAAE,MAAM,CAAC,EAAE,IAAI,CAAC,IAAI,SAAS,CAAC;IAChF,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,WAAW,EAAE,SAAS,EAAE,MAAM,CAAC,EAAE,IAAI,CAAC,IAAI,SAAS,CAAC;IAC3E,MAAM,QAAQ,GAAG,MAAM,CAAC,CAAC,QAAQ,EAAE,aAAa,CAAC,EAAE,IAAI,CAAC,CAAC;IACzD,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,QAAQ,EAAE,SAAS,EAAE,QAAQ,CAAC,EAAE,IAAI,CAAC,IAAI,SAAS,CAAC;IAE1E,OAAO;QACL,SAAS,EAAE,IAAI;QACf,MAAM;QACN,MAAM;QACN,KAAK,EAAE,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS;QACjD,MAAM;KACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,8 @@
1
+ export type EnvFileInfo = {
2
+ found: string[];
3
+ hasDotEnv: boolean;
4
+ };
5
+ export declare function detectEnvFiles(root?: string): EnvFileInfo;
6
+ export declare function detectScripts(root?: string): Record<string, string>;
7
+ export declare function detectNodeEngine(root?: string): string | undefined;
8
+ //# sourceMappingURL=meta.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"meta.d.ts","sourceRoot":"","sources":["../../src/detect/meta.ts"],"names":[],"mappings":"AAUA,MAAM,MAAM,WAAW,GAAG;IACxB,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,SAAS,EAAE,OAAO,CAAC;CACpB,CAAC;AAEF,wBAAgB,cAAc,CAAC,IAAI,SAAgB,GAAG,WAAW,CAOhE;AAED,wBAAgB,aAAa,CAAC,IAAI,SAAgB,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAS1E;AAED,wBAAgB,gBAAgB,CAAC,IAAI,SAAgB,GAAG,MAAM,GAAG,SAAS,CASzE"}
@@ -0,0 +1,42 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ const CANDIDATES = [
4
+ '.env.example',
5
+ '.env.sample',
6
+ '.env.template',
7
+ 'env.example',
8
+ ];
9
+ export function detectEnvFiles(root = process.cwd()) {
10
+ const found = [];
11
+ for (const name of CANDIDATES) {
12
+ if (fs.existsSync(path.join(root, name)))
13
+ found.push(name);
14
+ }
15
+ const hasDotEnv = fs.existsSync(path.join(root, '.env'));
16
+ return { found, hasDotEnv };
17
+ }
18
+ export function detectScripts(root = process.cwd()) {
19
+ const file = path.join(root, 'package.json');
20
+ if (!fs.existsSync(file))
21
+ return {};
22
+ try {
23
+ const pkg = JSON.parse(fs.readFileSync(file, 'utf8'));
24
+ return pkg.scripts || {};
25
+ }
26
+ catch {
27
+ return {};
28
+ }
29
+ }
30
+ export function detectNodeEngine(root = process.cwd()) {
31
+ const file = path.join(root, 'package.json');
32
+ if (!fs.existsSync(file))
33
+ return undefined;
34
+ try {
35
+ const pkg = JSON.parse(fs.readFileSync(file, 'utf8'));
36
+ return pkg.engines?.node;
37
+ }
38
+ catch {
39
+ return undefined;
40
+ }
41
+ }
42
+ //# sourceMappingURL=meta.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"meta.js","sourceRoot":"","sources":["../../src/detect/meta.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,SAAS,CAAC;AAC9B,OAAO,KAAK,IAAI,MAAM,WAAW,CAAC;AAElC,MAAM,UAAU,GAAG;IACjB,cAAc;IACd,aAAa;IACb,eAAe;IACf,aAAa;CACd,CAAC;AAOF,MAAM,UAAU,cAAc,CAAC,IAAI,GAAG,OAAO,CAAC,GAAG,EAAE;IACjD,MAAM,KAAK,GAAa,EAAE,CAAC;IAC3B,KAAK,MAAM,IAAI,IAAI,UAAU,EAAE,CAAC;QAC9B,IAAI,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC7D,CAAC;IACD,MAAM,SAAS,GAAG,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,CAAC;IACzD,OAAO,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC;AAC9B,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,IAAI,GAAG,OAAO,CAAC,GAAG,EAAE;IAChD,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;IAC7C,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC;QAAE,OAAO,EAAE,CAAC;IACpC,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,CAAyC,CAAC;QAC9F,OAAO,GAAG,CAAC,OAAO,IAAI,EAAE,CAAC;IAC3B,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,IAAI,GAAG,OAAO,CAAC,GAAG,EAAE;IACnD,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;IAC7C,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC;QAAE,OAAO,SAAS,CAAC;IAC3C,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,CAAoC,CAAC;QACzF,OAAO,GAAG,CAAC,OAAO,EAAE,IAAI,CAAC;IAC3B,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,SAAS,CAAC;IACnB,CAAC;AACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function detectRoutes(stackId: string, root?: string): string[];
2
+ //# sourceMappingURL=routes.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes.d.ts","sourceRoot":"","sources":["../../src/detect/routes.ts"],"names":[],"mappings":"AAwFA,wBAAgB,YAAY,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,SAAgB,GAAG,MAAM,EAAE,CAQ5E"}
@@ -0,0 +1,104 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ const IGNORE = new Set(['node_modules', '.git', 'dist', 'build', '.next', 'coverage', '.turbo']);
4
+ function walkFiles(dir, maxDepth = 6, depth = 0) {
5
+ if (depth > maxDepth)
6
+ return [];
7
+ let entries;
8
+ try {
9
+ entries = fs.readdirSync(dir, { withFileTypes: true });
10
+ }
11
+ catch {
12
+ return [];
13
+ }
14
+ const files = [];
15
+ for (const ent of entries) {
16
+ if (IGNORE.has(ent.name))
17
+ continue;
18
+ const full = path.join(dir, ent.name);
19
+ if (ent.isDirectory()) {
20
+ files.push(...walkFiles(full, maxDepth, depth + 1));
21
+ }
22
+ else if (ent.isFile()) {
23
+ files.push(full);
24
+ }
25
+ }
26
+ return files;
27
+ }
28
+ function nextAppRoutes(root) {
29
+ const routes = [];
30
+ const appDir = path.join(root, 'app');
31
+ if (!fs.existsSync(appDir))
32
+ return routes;
33
+ for (const file of walkFiles(appDir, 8)) {
34
+ const rel = path.relative(appDir, file).replace(/\\/g, '/');
35
+ if (rel.endsWith('/page.tsx') || rel.endsWith('/page.ts') || rel.endsWith('/page.jsx') || rel.endsWith('/page.js')) {
36
+ const routePath = '/' + rel.replace(/\/page\.(tsx|ts|jsx|js)$/, '').replace(/^page\.(tsx|ts|jsx|js)$/, '');
37
+ routes.push(routePath === '/' ? '/' : routePath.replace(/\/index$/, '') || '/');
38
+ }
39
+ }
40
+ return [...new Set(routes)].sort();
41
+ }
42
+ function nextPagesRoutes(root) {
43
+ const pagesDir = path.join(root, 'pages');
44
+ if (!fs.existsSync(pagesDir))
45
+ return [];
46
+ const routes = [];
47
+ for (const file of walkFiles(pagesDir, 6)) {
48
+ if (!/\.(tsx|ts|jsx|js)$/.test(file))
49
+ continue;
50
+ const rel = path.relative(pagesDir, file).replace(/\\/g, '/');
51
+ if (rel.startsWith('api/'))
52
+ continue;
53
+ const withoutExt = rel.replace(/\.(tsx|ts|jsx|js)$/, '');
54
+ if (withoutExt === 'index')
55
+ routes.push('/');
56
+ else
57
+ routes.push('/' + withoutExt.replace(/\/index$/, ''));
58
+ }
59
+ return [...new Set(routes)].sort();
60
+ }
61
+ const ROUTE_PATTERNS = [
62
+ /\.(?:get|post|put|patch|delete|all)\(\s*['"`]([^'"`]+)['"`]/gi,
63
+ /@(?:Get|Post|Put|Patch|Delete)\(\s*['"`]([^'"`]+)['"`]/g,
64
+ /router\.(?:get|post|put|patch|delete|use)\(\s*['"`]([^'"`]+)['"`]/gi,
65
+ /app\.(?:get|post|put|patch|delete|use)\(\s*['"`]([^'"`]+)['"`]/gi,
66
+ ];
67
+ function scanSourceRoutes(root) {
68
+ const srcDirs = ['src', 'server', 'api', 'routes', 'app'].map((d) => path.join(root, d)).filter((d) => fs.existsSync(d));
69
+ const routes = new Set();
70
+ for (const dir of srcDirs) {
71
+ for (const file of walkFiles(dir, 5)) {
72
+ if (!/\.(tsx?|jsx?)$/.test(file))
73
+ continue;
74
+ let content;
75
+ try {
76
+ content = fs.readFileSync(file, 'utf8');
77
+ }
78
+ catch {
79
+ continue;
80
+ }
81
+ for (const re of ROUTE_PATTERNS) {
82
+ re.lastIndex = 0;
83
+ let m;
84
+ while ((m = re.exec(content))) {
85
+ const r = m[1]?.trim();
86
+ if (r && r.length < 120)
87
+ routes.add(r.startsWith('/') ? r : `/${r}`);
88
+ }
89
+ }
90
+ }
91
+ }
92
+ return [...routes].sort();
93
+ }
94
+ export function detectRoutes(stackId, root = process.cwd()) {
95
+ if (stackId === 'nextjs') {
96
+ const app = nextAppRoutes(root);
97
+ const pages = nextPagesRoutes(root);
98
+ const merged = [...new Set([...app, ...pages])];
99
+ if (merged.length)
100
+ return merged.slice(0, 200);
101
+ }
102
+ return scanSourceRoutes(root).slice(0, 200);
103
+ }
104
+ //# sourceMappingURL=routes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes.js","sourceRoot":"","sources":["../../src/detect/routes.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ export type DetectedStack = {
2
+ id: string;
3
+ label: string;
4
+ confidence: 'high' | 'medium' | 'low';
5
+ hints: string[];
6
+ frameworkVersions: Record<string, string>;
7
+ };
8
+ export declare function detectStack(root?: string): DetectedStack;
9
+ export declare function readProjectName(root?: string): string;
10
+ export declare function listDependencies(root?: string): {
11
+ name: string;
12
+ version: string;
13
+ dev: boolean;
14
+ }[];
15
+ //# sourceMappingURL=stack.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stack.d.ts","sourceRoot":"","sources":["../../src/detect/stack.ts"],"names":[],"mappings":"AAGA,MAAM,MAAM,aAAa,GAAG;IAC1B,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,GAAG,QAAQ,GAAG,KAAK,CAAC;IACtC,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,iBAAiB,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CAC3C,CAAC;AAiDF,wBAAgB,WAAW,CAAC,IAAI,SAAgB,GAAG,aAAa,CA4D/D;AAED,wBAAgB,eAAe,CAAC,IAAI,SAAgB,GAAG,MAAM,CAI5D;AAED,wBAAgB,gBAAgB,CAAC,IAAI,SAAgB,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,MAAM,CAAC;IAAC,GAAG,EAAE,OAAO,CAAA;CAAE,EAAE,CAWxG"}