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,65 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ import { ok, title, info } from '../utils/log.js';
4
+ const WORKFLOW = `# Generated by sublyzer-snapshot ci — commit or customize as needed
5
+ name: Sublyzer Snapshot
6
+
7
+ on:
8
+ push:
9
+ branches: [main, master]
10
+ pull_request:
11
+ schedule:
12
+ - cron: '0 9 * * 1' # Weekly Monday 09:00 UTC
13
+
14
+ jobs:
15
+ snapshot:
16
+ runs-on: ubuntu-latest
17
+ steps:
18
+ - uses: actions/checkout@v4
19
+
20
+ - uses: actions/setup-node@v4
21
+ with:
22
+ node-version: '20'
23
+ cache: npm
24
+
25
+ - name: Install Sublyzer Snapshot
26
+ run: npm install -g sublyzer-snapshot@latest
27
+
28
+ - name: Link integration
29
+ env:
30
+ SUBLYZER_INTEGRATION_CODE: \${{ secrets.SUBLYZER_INTEGRATION_CODE }}
31
+ run: sublyzer-snapshot init --code "$SUBLYZER_INTEGRATION_CODE" -y
32
+
33
+ - name: Scan and push
34
+ run: sublyzer-snapshot run --fail-on high --json > snapshot-result.json
35
+
36
+ - name: Upload report
37
+ run: sublyzer-snapshot report --out sublyzer-report.md
38
+
39
+ - uses: actions/upload-artifact@v4
40
+ if: always()
41
+ with:
42
+ name: sublyzer-snapshot
43
+ path: |
44
+ snapshot-result.json
45
+ sublyzer-report.md
46
+ `;
47
+ export async function runCi(opts = {}) {
48
+ const outPath = path.resolve(process.cwd(), opts.out || '.github/workflows/sublyzer-snapshot.yml');
49
+ const content = WORKFLOW;
50
+ if (opts.print !== false && !opts.out) {
51
+ title('Sublyzer Snapshot — CI template');
52
+ console.log(content);
53
+ info('Save to .github/workflows/sublyzer-snapshot.yml and add secret SUBLYZER_INTEGRATION_CODE');
54
+ return { path: outPath, content };
55
+ }
56
+ fs.mkdirSync(path.dirname(outPath), { recursive: true });
57
+ if (fs.existsSync(outPath)) {
58
+ throw new Error(`File already exists: ${outPath} (use --out to pick another path)`);
59
+ }
60
+ fs.writeFileSync(outPath, content, 'utf8');
61
+ ok(`Wrote ${outPath}`);
62
+ info('Add GitHub secret: SUBLYZER_INTEGRATION_CODE');
63
+ return { path: outPath, content };
64
+ }
65
+ //# sourceMappingURL=ci.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ci.js","sourceRoot":"","sources":["../../src/commands/ci.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,SAAS,CAAC;AAC9B,OAAO,KAAK,IAAI,MAAM,WAAW,CAAC;AAClC,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AAElD,MAAM,QAAQ,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA0ChB,CAAC;AAOF,MAAM,CAAC,KAAK,UAAU,KAAK,CAAC,OAAkB,EAAE;IAC9C,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,GAAG,IAAI,yCAAyC,CAAC,CAAC;IACnG,MAAM,OAAO,GAAG,QAAQ,CAAC;IAEzB,IAAI,IAAI,CAAC,KAAK,KAAK,KAAK,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC;QACtC,KAAK,CAAC,iCAAiC,CAAC,CAAC;QACzC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QACrB,IAAI,CAAC,0FAA0F,CAAC,CAAC;QACjG,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC;IACpC,CAAC;IAED,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IACzD,IAAI,EAAE,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;QAC3B,MAAM,IAAI,KAAK,CAAC,wBAAwB,OAAO,mCAAmC,CAAC,CAAC;IACtF,CAAC;IACD,EAAE,CAAC,aAAa,CAAC,OAAO,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;IAC3C,EAAE,CAAC,SAAS,OAAO,EAAE,CAAC,CAAC;IACvB,IAAI,CAAC,8CAA8C,CAAC,CAAC;IACrD,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC;AACpC,CAAC"}
@@ -0,0 +1,7 @@
1
+ export type CompareOptions = {
2
+ json?: boolean;
3
+ rescan?: boolean;
4
+ skipAudit?: boolean;
5
+ };
6
+ export declare function runCompare(opts?: CompareOptions): Promise<Record<string, unknown>>;
7
+ //# sourceMappingURL=compare.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"compare.d.ts","sourceRoot":"","sources":["../../src/commands/compare.ts"],"names":[],"mappings":"AASA,MAAM,MAAM,cAAc,GAAG;IAC3B,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB,CAAC;AAEF,wBAAsB,UAAU,CAAC,IAAI,GAAE,cAAmB,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CA8D5F"}
@@ -0,0 +1,60 @@
1
+ import { loadConfig } from '../config.js';
2
+ import { diffSnapshots, loadLastSnapshot, loadPreviousSnapshot, } from '../scan/history.js';
3
+ import { buildProjectSnapshot } from '../scan/snapshot.js';
4
+ import { info, title } from '../utils/log.js';
5
+ export async function runCompare(opts = {}) {
6
+ const config = loadConfig();
7
+ const root = config.projectRoot || process.cwd();
8
+ const previous = loadPreviousSnapshot(root) || loadLastSnapshot(root);
9
+ const current = opts.rescan
10
+ ? buildProjectSnapshot(root, { skipAudit: opts.skipAudit })
11
+ : loadLastSnapshot(root);
12
+ if (!current) {
13
+ throw new Error('No scan data. Run: sublyzer-snapshot run');
14
+ }
15
+ if (!previous || previous.scannedAt === current.scannedAt) {
16
+ throw new Error('Need at least two scans to compare. Run `sublyzer-snapshot run` again.');
17
+ }
18
+ const healthDelta = (current.health?.score ?? current.summary.healthScore) -
19
+ (previous.health?.score ?? previous.summary?.healthScore ?? 0);
20
+ const diff = diffSnapshots(previous, current, healthDelta);
21
+ const payload = {
22
+ previous: {
23
+ scannedAt: previous.scannedAt,
24
+ healthScore: previous.health?.score ?? previous.summary?.healthScore,
25
+ routes: previous.summary.routeCount,
26
+ vulnerabilities: previous.summary.vulnerablePackages,
27
+ },
28
+ current: {
29
+ scannedAt: current.scannedAt,
30
+ healthScore: current.health?.score ?? current.summary.healthScore,
31
+ routes: current.summary.routeCount,
32
+ vulnerabilities: current.summary.vulnerablePackages,
33
+ },
34
+ diff,
35
+ };
36
+ if (opts.json)
37
+ return payload;
38
+ title('Sublyzer Snapshot — compare');
39
+ info(`Previous: ${previous.scannedAt} → health ${previous.health?.score ?? '?'}/100`);
40
+ info(`Current: ${current.scannedAt} → health ${current.health?.score ?? '?'}/100`);
41
+ console.log('');
42
+ if (healthDelta !== 0) {
43
+ console.log(` Health score: ${healthDelta > 0 ? '+' : ''}${healthDelta}`);
44
+ }
45
+ console.log(` Vulnerabilities: ${diff.vulnDelta.total >= 0 ? '+' : ''}${diff.vulnDelta.total} (C:${diff.vulnDelta.critical >= 0 ? '+' : ''}${diff.vulnDelta.critical} H:${diff.vulnDelta.high >= 0 ? '+' : ''}${diff.vulnDelta.high})`);
46
+ console.log(` Dependencies: ${diff.depDelta >= 0 ? '+' : ''}${diff.depDelta}`);
47
+ if (diff.routesAdded.length) {
48
+ console.log(` Routes added (${diff.routesAdded.length}):`);
49
+ for (const r of diff.routesAdded.slice(0, 12))
50
+ console.log(` + ${r}`);
51
+ }
52
+ if (diff.routesRemoved.length) {
53
+ console.log(` Routes removed (${diff.routesRemoved.length}):`);
54
+ for (const r of diff.routesRemoved.slice(0, 12))
55
+ console.log(` - ${r}`);
56
+ }
57
+ console.log('');
58
+ return payload;
59
+ }
60
+ //# sourceMappingURL=compare.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"compare.js","sourceRoot":"","sources":["../../src/commands/compare.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ export type DoctorOptions = {
2
+ json?: boolean;
3
+ };
4
+ export type DoctorCheck = {
5
+ name: string;
6
+ ok: boolean;
7
+ message: string;
8
+ };
9
+ export declare function runDoctor(opts?: DoctorOptions): Promise<{
10
+ ok: boolean;
11
+ checks: DoctorCheck[];
12
+ }>;
13
+ //# sourceMappingURL=doctor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"doctor.d.ts","sourceRoot":"","sources":["../../src/commands/doctor.ts"],"names":[],"mappings":"AAMA,MAAM,MAAM,aAAa,GAAG;IAAE,IAAI,CAAC,EAAE,OAAO,CAAA;CAAE,CAAC;AAE/C,MAAM,MAAM,WAAW,GAAG;IACxB,IAAI,EAAE,MAAM,CAAC;IACb,EAAE,EAAE,OAAO,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,wBAAsB,SAAS,CAAC,IAAI,GAAE,aAAkB,GAAG,OAAO,CAAC;IAAE,EAAE,EAAE,OAAO,CAAC;IAAC,MAAM,EAAE,WAAW,EAAE,CAAA;CAAE,CAAC,CA+FzG"}
@@ -0,0 +1,100 @@
1
+ import { spawnSync } from 'node:child_process';
2
+ import { configExists, loadConfig, resolveReadKey } from '../config.js';
3
+ import { fetchPublicSnapshot, validateIntegrationCode } from '../api/sublyzer.js';
4
+ import { detectStack } from '../detect/stack.js';
5
+ import { info, ok, title, warn } from '../utils/log.js';
6
+ export async function runDoctor(opts = {}) {
7
+ const checks = [];
8
+ // Config
9
+ if (!configExists()) {
10
+ checks.push({ name: 'config', ok: false, message: 'Missing .sublyzer/snapshot.json — run init' });
11
+ }
12
+ else {
13
+ checks.push({ name: 'config', ok: true, message: 'Config file found' });
14
+ }
15
+ let config;
16
+ try {
17
+ config = loadConfig();
18
+ }
19
+ catch (e) {
20
+ checks.push({ name: 'config-parse', ok: false, message: e?.message || 'Invalid config' });
21
+ }
22
+ // Node version
23
+ const nodeMajor = parseInt(process.versions.node.split('.')[0] || '0', 10);
24
+ checks.push({
25
+ name: 'node',
26
+ ok: nodeMajor >= 18,
27
+ message: nodeMajor >= 18 ? `Node ${process.versions.node}` : `Node ${process.versions.node} — require 18+`,
28
+ });
29
+ // npm
30
+ const npm = spawnSync(process.platform === 'win32' ? 'npm.cmd' : 'npm', ['--version'], { encoding: 'utf8' });
31
+ checks.push({
32
+ name: 'npm',
33
+ ok: true,
34
+ message: npm.status === 0 ? `npm ${(npm.stdout || '').trim()}` : 'npm not found — audit will be skipped on run',
35
+ });
36
+ // Stack
37
+ const stack = detectStack(config?.projectRoot || process.cwd());
38
+ checks.push({
39
+ name: 'stack',
40
+ ok: stack.id !== 'unknown',
41
+ message: stack.id !== 'unknown' ? `${stack.label} (${stack.confidence})` : 'Could not detect stack',
42
+ });
43
+ if (config) {
44
+ try {
45
+ const validation = await validateIntegrationCode(config.apiUrl, config.integrationCode);
46
+ checks.push({
47
+ name: 'integration',
48
+ ok: Boolean(validation.valid),
49
+ message: validation.valid
50
+ ? `${validation.integration?.name} (${validation.integration?.status})`
51
+ : validation.message || 'Invalid integration code',
52
+ });
53
+ }
54
+ catch (e) {
55
+ checks.push({ name: 'integration', ok: false, message: e?.message || 'API unreachable' });
56
+ }
57
+ const readKey = resolveReadKey(config);
58
+ if (readKey) {
59
+ try {
60
+ const pull = await fetchPublicSnapshot(config.apiUrl, config.integrationCode, readKey, {
61
+ limit: 1,
62
+ include: ['stats'],
63
+ });
64
+ checks.push({
65
+ name: 'read-key',
66
+ ok: pull.ok,
67
+ message: pull.ok ? 'Read key valid — pull enabled' : pull.error || `HTTP ${pull.status}`,
68
+ });
69
+ }
70
+ catch (e) {
71
+ checks.push({ name: 'read-key', ok: false, message: e?.message || 'Read key check failed' });
72
+ }
73
+ }
74
+ else {
75
+ checks.push({
76
+ name: 'read-key',
77
+ ok: true,
78
+ message: 'Not configured (optional — enables pull)',
79
+ });
80
+ }
81
+ }
82
+ const allOk = checks.every((c) => c.ok);
83
+ if (opts.json) {
84
+ return { ok: allOk, checks };
85
+ }
86
+ title('Sublyzer Snapshot — doctor');
87
+ for (const c of checks) {
88
+ if (c.ok)
89
+ ok(`${c.name}: ${c.message}`);
90
+ else
91
+ warn(`${c.name}: ${c.message}`);
92
+ }
93
+ console.log('');
94
+ if (allOk)
95
+ ok('All required checks passed');
96
+ else
97
+ info('Fix issues above, then re-run doctor');
98
+ return { ok: allOk, checks };
99
+ }
100
+ //# sourceMappingURL=doctor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"doctor.js","sourceRoot":"","sources":["../../src/commands/doctor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ export type InitOptions = {
2
+ code?: string;
3
+ readKey?: string;
4
+ apiUrl?: string;
5
+ dashboardUrl?: string;
6
+ yes?: boolean;
7
+ skipGitignore?: boolean;
8
+ };
9
+ export declare function runInit(opts?: InitOptions): Promise<void>;
10
+ //# sourceMappingURL=init.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../../src/commands/init.ts"],"names":[],"mappings":"AAaA,MAAM,MAAM,WAAW,GAAG;IACxB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,GAAG,CAAC,EAAE,OAAO,CAAC;IACd,aAAa,CAAC,EAAE,OAAO,CAAC;CACzB,CAAC;AAEF,wBAAsB,OAAO,CAAC,IAAI,GAAE,WAAgB,GAAG,OAAO,CAAC,IAAI,CAAC,CAyFnE"}
@@ -0,0 +1,88 @@
1
+ import * as path from 'node:path';
2
+ import { DEFAULT_API_URL, DEFAULT_DASHBOARD_URL, INTEGRATION_CODE_RE, } from '../constants.js';
3
+ import { saveConfig } from '../config.js';
4
+ import { detectStack, readProjectName } from '../detect/stack.js';
5
+ import { validateIntegrationCode } from '../api/sublyzer.js';
6
+ import { promptOptional, promptRequired } from '../utils/prompt.js';
7
+ import { ensureGitignore, printSdkHint } from '../utils/gitignore.js';
8
+ import { info, ok, title } from '../utils/log.js';
9
+ export async function runInit(opts = {}) {
10
+ title('Sublyzer Snapshot — init');
11
+ const root = process.cwd();
12
+ const stack = detectStack(root);
13
+ const projectName = readProjectName(root);
14
+ info(`Project: ${projectName}`);
15
+ info(`Detected stack: ${stack.label} (${stack.confidence})`);
16
+ if (stack.hints.length) {
17
+ for (const h of stack.hints)
18
+ info(` hint: ${h}`);
19
+ }
20
+ const apiUrl = (opts.apiUrl || process.env.SUBLYZER_API_URL || DEFAULT_API_URL).replace(/\/$/, '');
21
+ const dashboardUrl = (opts.dashboardUrl || process.env.SUBLYZER_DASHBOARD_URL || DEFAULT_DASHBOARD_URL).replace(/\/$/, '');
22
+ let code = (opts.code || process.env.SUBLYZER_INTEGRATION_CODE || process.env.SUBLYZER_CODE || '')
23
+ .trim()
24
+ .toUpperCase();
25
+ if (!code) {
26
+ console.log('');
27
+ console.log(' Get your integration code at: https://sublyzer.com/dashboard');
28
+ console.log(' (Integrations → copy the 24-character code)');
29
+ console.log('');
30
+ code = (await promptRequired('Integration code (24 chars)', (v) => {
31
+ const c = v.trim().toUpperCase();
32
+ if (!INTEGRATION_CODE_RE.test(c))
33
+ return 'Code must be 24 uppercase letters/digits (A-Z, 0-9)';
34
+ return null;
35
+ })).trim().toUpperCase();
36
+ }
37
+ else if (!INTEGRATION_CODE_RE.test(code)) {
38
+ throw new Error('Invalid integration code format (expected 24 uppercase alphanumeric characters)');
39
+ }
40
+ let readKey = (opts.readKey || process.env.SUBLYZER_READ_KEY || '').trim();
41
+ if (!readKey && !opts.yes) {
42
+ console.log('');
43
+ console.log(' Optional: apiReadKey enables `sublyzer-snapshot pull` (read data back).');
44
+ console.log(' Get it from: GET /integrations/{id}/read-access (dashboard JWT) or skip for now.');
45
+ console.log('');
46
+ readKey = (await promptOptional('Read key (optional, press Enter to skip)')) || '';
47
+ }
48
+ info(`Validating code against ${apiUrl}…`);
49
+ const validation = await validateIntegrationCode(apiUrl, code);
50
+ if (!validation.valid || !validation.integration) {
51
+ throw new Error(validation.message || 'Integration code not found or invalid');
52
+ }
53
+ ok(`Integration "${validation.integration.name}" (${validation.integration.status})`);
54
+ const now = new Date().toISOString();
55
+ const config = {
56
+ version: 1,
57
+ integrationCode: code,
58
+ apiUrl,
59
+ dashboardUrl,
60
+ integrationId: validation.integration.id,
61
+ integrationName: validation.integration.name,
62
+ projectName,
63
+ projectRoot: root,
64
+ stack: stack.id,
65
+ createdAt: now,
66
+ updatedAt: now,
67
+ };
68
+ if (readKey.length >= 32) {
69
+ config.readKey = readKey;
70
+ ok('Read key saved — `pull` command enabled');
71
+ }
72
+ saveConfig(config, root);
73
+ ok(`Wrote ${path.join('.sublyzer', 'snapshot.json')}`);
74
+ if (!opts.skipGitignore) {
75
+ ensureGitignore(root);
76
+ }
77
+ console.log('');
78
+ console.log(' Next steps:');
79
+ console.log(' sublyzer-snapshot run Scan + push to Sublyzer');
80
+ console.log(' sublyzer-snapshot report Markdown health report');
81
+ console.log(' sublyzer-snapshot compare Diff vs previous scan');
82
+ console.log(' sublyzer-snapshot ci GitHub Actions template');
83
+ console.log(' sublyzer-snapshot doctor Verify setup');
84
+ console.log(' sublyzer-snapshot open Open dashboard');
85
+ console.log('');
86
+ printSdkHint(stack.id, code, apiUrl);
87
+ }
88
+ //# sourceMappingURL=init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.js","sourceRoot":"","sources":["../../src/commands/init.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function runOpen(): Promise<void>;
2
+ //# sourceMappingURL=open.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"open.d.ts","sourceRoot":"","sources":["../../src/commands/open.ts"],"names":[],"mappings":"AAIA,wBAAsB,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC,CAc7C"}
@@ -0,0 +1,16 @@
1
+ import { spawn } from 'node:child_process';
2
+ import { dashboardIntegrationUrl, loadConfig } from '../config.js';
3
+ import { info, ok } from '../utils/log.js';
4
+ export async function runOpen() {
5
+ const config = loadConfig();
6
+ const url = dashboardIntegrationUrl(config);
7
+ info(`Opening ${url}`);
8
+ const cmd = process.platform === 'win32'
9
+ ? spawn('cmd', ['/c', 'start', '', url], { detached: true, stdio: 'ignore' })
10
+ : process.platform === 'darwin'
11
+ ? spawn('open', [url], { detached: true, stdio: 'ignore' })
12
+ : spawn('xdg-open', [url], { detached: true, stdio: 'ignore' });
13
+ cmd.unref();
14
+ ok('Browser launched (or copy the URL above)');
15
+ }
16
+ //# sourceMappingURL=open.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"open.js","sourceRoot":"","sources":["../../src/commands/open.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,uBAAuB,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AACnE,OAAO,EAAE,IAAI,EAAE,EAAE,EAAE,MAAM,iBAAiB,CAAC;AAE3C,MAAM,CAAC,KAAK,UAAU,OAAO;IAC3B,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,GAAG,GAAG,uBAAuB,CAAC,MAAM,CAAC,CAAC;IAC5C,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC,CAAC;IAEvB,MAAM,GAAG,GACP,OAAO,CAAC,QAAQ,KAAK,OAAO;QAC1B,CAAC,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,IAAI,EAAE,OAAO,EAAE,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC;QAC7E,CAAC,CAAC,OAAO,CAAC,QAAQ,KAAK,QAAQ;YAC7B,CAAC,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC;YAC3D,CAAC,CAAC,KAAK,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAC;IAEtE,GAAG,CAAC,KAAK,EAAE,CAAC;IACZ,EAAE,CAAC,0CAA0C,CAAC,CAAC;AACjD,CAAC"}
@@ -0,0 +1,9 @@
1
+ export type PullOptions = {
2
+ readKey?: string;
3
+ limit?: number;
4
+ windowDays?: number;
5
+ include?: string;
6
+ json?: boolean;
7
+ };
8
+ export declare function runPull(opts?: PullOptions): Promise<unknown>;
9
+ //# sourceMappingURL=pull.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pull.d.ts","sourceRoot":"","sources":["../../src/commands/pull.ts"],"names":[],"mappings":"AAIA,MAAM,MAAM,WAAW,GAAG;IACxB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB,CAAC;AAEF,wBAAsB,OAAO,CAAC,IAAI,GAAE,WAAgB,GAAG,OAAO,CAAC,OAAO,CAAC,CAuCtE"}
@@ -0,0 +1,34 @@
1
+ import { loadConfig, resolveReadKey } from '../config.js';
2
+ import { fetchPublicSnapshot } from '../api/sublyzer.js';
3
+ import { info, ok, title } from '../utils/log.js';
4
+ export async function runPull(opts = {}) {
5
+ const config = loadConfig();
6
+ const readKey = resolveReadKey(config, opts.readKey);
7
+ if (!readKey || readKey.length < 32) {
8
+ throw new Error('Read key required. Set SUBLYZER_READ_KEY, save in config via init, or pass --read-key. ' +
9
+ 'Get it from GET /integrations/{id}/read-access (dashboard JWT).');
10
+ }
11
+ const include = opts.include
12
+ ? opts.include.split(',').map((s) => s.trim()).filter(Boolean)
13
+ : ['stats', 'events'];
14
+ if (!opts.json) {
15
+ title('Sublyzer Snapshot — pull');
16
+ info(`Fetching from ${config.apiUrl}…`);
17
+ }
18
+ const result = await fetchPublicSnapshot(config.apiUrl, config.integrationCode, readKey, {
19
+ limit: opts.limit ?? 50,
20
+ windowDays: opts.windowDays ?? 7,
21
+ include,
22
+ });
23
+ if (!result.ok) {
24
+ throw new Error(result.error || `Pull failed (HTTP ${result.status})`);
25
+ }
26
+ if (opts.json) {
27
+ return result.data;
28
+ }
29
+ ok('Data fetched');
30
+ console.log(JSON.stringify(result.data, null, 2));
31
+ console.log('');
32
+ return result.data;
33
+ }
34
+ //# sourceMappingURL=pull.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pull.js","sourceRoot":"","sources":["../../src/commands/pull.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAC1D,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAC;AACzD,OAAO,EAAE,IAAI,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAUlD,MAAM,CAAC,KAAK,UAAU,OAAO,CAAC,OAAoB,EAAE;IAClD,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,OAAO,GAAG,cAAc,CAAC,MAAM,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IAErD,IAAI,CAAC,OAAO,IAAI,OAAO,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;QACpC,MAAM,IAAI,KAAK,CACb,yFAAyF;YACvF,iEAAiE,CACpE,CAAC;IACJ,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO;QAC1B,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC;QAC9D,CAAC,CAAC,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;IAExB,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;QACf,KAAK,CAAC,0BAA0B,CAAC,CAAC;QAClC,IAAI,CAAC,iBAAiB,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC;IAC1C,CAAC;IAED,MAAM,MAAM,GAAG,MAAM,mBAAmB,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,eAAe,EAAE,OAAO,EAAE;QACvF,KAAK,EAAE,IAAI,CAAC,KAAK,IAAI,EAAE;QACvB,UAAU,EAAE,IAAI,CAAC,UAAU,IAAI,CAAC;QAChC,OAAO;KACR,CAAC,CAAC;IAEH,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;QACf,MAAM,IAAI,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,qBAAqB,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC;IACzE,CAAC;IAED,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;QACd,OAAO,MAAM,CAAC,IAAI,CAAC;IACrB,CAAC;IAED,EAAE,CAAC,cAAc,CAAC,CAAC;IACnB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IAClD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAEhB,OAAO,MAAM,CAAC,IAAI,CAAC;AACrB,CAAC"}
@@ -0,0 +1,8 @@
1
+ export type ReportOptions = {
2
+ out?: string;
3
+ rescan?: boolean;
4
+ skipAudit?: boolean;
5
+ json?: boolean;
6
+ };
7
+ export declare function runReport(opts?: ReportOptions): Promise<string>;
8
+ //# sourceMappingURL=report.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"report.d.ts","sourceRoot":"","sources":["../../src/commands/report.ts"],"names":[],"mappings":"AASA,MAAM,MAAM,aAAa,GAAG;IAC1B,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB,CAAC;AAEF,wBAAsB,SAAS,CAAC,IAAI,GAAE,aAAkB,GAAG,OAAO,CAAC,MAAM,CAAC,CAiDzE"}
@@ -0,0 +1,47 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ import { dashboardIntegrationUrl, loadConfig } from '../config.js';
4
+ import { diffSnapshots, loadLastSnapshot, loadPreviousSnapshot } from '../scan/history.js';
5
+ import { buildProjectSnapshot } from '../scan/snapshot.js';
6
+ import { computeHealthScore } from '../scan/health-score.js';
7
+ import { renderMarkdownReport } from '../report/markdown.js';
8
+ import { info, ok, title } from '../utils/log.js';
9
+ export async function runReport(opts = {}) {
10
+ const config = loadConfig();
11
+ const root = config.projectRoot || process.cwd();
12
+ let snapshot = opts.rescan
13
+ ? buildProjectSnapshot(root, { skipAudit: opts.skipAudit })
14
+ : loadLastSnapshot(root);
15
+ if (!snapshot) {
16
+ if (!opts.rescan) {
17
+ info('No cached scan — running fresh scan…');
18
+ snapshot = buildProjectSnapshot(root, { skipAudit: opts.skipAudit });
19
+ }
20
+ else {
21
+ throw new Error('Scan failed');
22
+ }
23
+ }
24
+ const health = snapshot.health ?? computeHealthScore(snapshot);
25
+ const previous = loadPreviousSnapshot(root);
26
+ const diff = previous
27
+ ? diffSnapshots(previous, snapshot, health.score - (previous.health?.score ?? previous.summary?.healthScore ?? 0))
28
+ : null;
29
+ const markdown = renderMarkdownReport(snapshot, health, diff, dashboardIntegrationUrl(config));
30
+ if (opts.json) {
31
+ return JSON.stringify({ markdown, health, snapshot: { scannedAt: snapshot.scannedAt, projectName: snapshot.projectName } }, null, 2);
32
+ }
33
+ if (opts.out) {
34
+ const outPath = path.resolve(root, opts.out);
35
+ fs.mkdirSync(path.dirname(outPath), { recursive: true });
36
+ fs.writeFileSync(outPath, markdown, 'utf8');
37
+ if (!opts.json) {
38
+ title('Sublyzer Snapshot — report');
39
+ ok(`Wrote ${outPath}`);
40
+ }
41
+ return markdown;
42
+ }
43
+ title('Sublyzer Snapshot — report');
44
+ console.log(markdown);
45
+ return markdown;
46
+ }
47
+ //# sourceMappingURL=report.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"report.js","sourceRoot":"","sources":["../../src/commands/report.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,21 @@
1
+ import type { FailOnLevel } from '../constants.js';
2
+ import { diffSnapshots } from '../scan/history.js';
3
+ import { type ProjectSnapshot } from '../scan/snapshot.js';
4
+ export type RunOptions = {
5
+ dryRun?: boolean;
6
+ skipAudit?: boolean;
7
+ skipOutdated?: boolean;
8
+ json?: boolean;
9
+ failOn?: FailOnLevel;
10
+ };
11
+ export type RunResult = {
12
+ success: boolean;
13
+ dryRun: boolean;
14
+ snapshot: ProjectSnapshot;
15
+ eventsSent?: number;
16
+ dashboardUrl?: string;
17
+ diff?: ReturnType<typeof diffSnapshots>;
18
+ policyFailed?: boolean;
19
+ };
20
+ export declare function runScan(opts?: RunOptions): Promise<RunResult>;
21
+ //# sourceMappingURL=run.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run.d.ts","sourceRoot":"","sources":["../../src/commands/run.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AACnD,OAAO,EACL,aAAa,EAGd,MAAM,oBAAoB,CAAC;AAE5B,OAAO,EAIL,KAAK,eAAe,EACrB,MAAM,qBAAqB,CAAC;AAG7B,MAAM,MAAM,UAAU,GAAG;IACvB,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,MAAM,CAAC,EAAE,WAAW,CAAC;CACtB,CAAC;AAEF,MAAM,MAAM,SAAS,GAAG;IACtB,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,EAAE,OAAO,CAAC;IAChB,QAAQ,EAAE,eAAe,CAAC;IAC1B,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,IAAI,CAAC,EAAE,UAAU,CAAC,OAAO,aAAa,CAAC,CAAC;IACxC,YAAY,CAAC,EAAE,OAAO,CAAC;CACxB,CAAC;AAuBF,wBAAsB,OAAO,CAAC,IAAI,GAAE,UAAe,GAAG,OAAO,CAAC,SAAS,CAAC,CAgFvE"}