@nepopsx/cli 0.0.3 → 0.0.5

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 (168) hide show
  1. package/dist/agents/configuration.d.ts +9 -0
  2. package/dist/agents/configuration.d.ts.map +1 -0
  3. package/dist/agents/configuration.js +57 -0
  4. package/dist/agents/configuration.js.map +1 -0
  5. package/dist/agents/remote-config.d.ts +30 -0
  6. package/dist/agents/remote-config.d.ts.map +1 -0
  7. package/dist/agents/remote-config.js +102 -0
  8. package/dist/agents/remote-config.js.map +1 -0
  9. package/dist/commands/activate.d.ts.map +1 -1
  10. package/dist/commands/activate.js +4 -3
  11. package/dist/commands/activate.js.map +1 -1
  12. package/dist/commands/decode.js +2 -2
  13. package/dist/commands/decode.js.map +1 -1
  14. package/dist/commands/doctor.js +16 -16
  15. package/dist/commands/doctor.js.map +1 -1
  16. package/dist/commands/init.d.ts.map +1 -1
  17. package/dist/commands/init.js +353 -18
  18. package/dist/commands/init.js.map +1 -1
  19. package/dist/commands/install.d.ts +1 -0
  20. package/dist/commands/install.d.ts.map +1 -1
  21. package/dist/commands/install.js +253 -25
  22. package/dist/commands/install.js.map +1 -1
  23. package/dist/commands/login.d.ts +6 -0
  24. package/dist/commands/login.d.ts.map +1 -0
  25. package/dist/commands/login.js +108 -0
  26. package/dist/commands/login.js.map +1 -0
  27. package/dist/commands/push-learnings.d.ts +14 -0
  28. package/dist/commands/push-learnings.d.ts.map +1 -0
  29. package/dist/commands/push-learnings.js +186 -0
  30. package/dist/commands/push-learnings.js.map +1 -0
  31. package/dist/commands/scan.d.ts +2 -0
  32. package/dist/commands/scan.d.ts.map +1 -0
  33. package/dist/commands/scan.js +520 -0
  34. package/dist/commands/scan.js.map +1 -0
  35. package/dist/commands/sync.d.ts +2 -0
  36. package/dist/commands/sync.d.ts.map +1 -1
  37. package/dist/commands/sync.js +203 -234
  38. package/dist/commands/sync.js.map +1 -1
  39. package/dist/config/api-config.d.ts.map +1 -1
  40. package/dist/config/api-config.js +4 -0
  41. package/dist/config/api-config.js.map +1 -1
  42. package/dist/constants.d.ts +7 -0
  43. package/dist/constants.d.ts.map +1 -0
  44. package/dist/constants.js +7 -0
  45. package/dist/constants.js.map +1 -0
  46. package/dist/customs/instructions.d.ts +3 -0
  47. package/dist/customs/instructions.d.ts.map +1 -0
  48. package/dist/customs/instructions.js +51 -0
  49. package/dist/customs/instructions.js.map +1 -0
  50. package/dist/enrichment/confirm.d.ts +17 -0
  51. package/dist/enrichment/confirm.d.ts.map +1 -0
  52. package/dist/enrichment/confirm.js +44 -0
  53. package/dist/enrichment/confirm.js.map +1 -0
  54. package/dist/enrichment/state.d.ts +8 -0
  55. package/dist/enrichment/state.d.ts.map +1 -0
  56. package/dist/enrichment/state.js +107 -0
  57. package/dist/enrichment/state.js.map +1 -0
  58. package/dist/enrichment/types.d.ts +13 -0
  59. package/dist/enrichment/types.d.ts.map +1 -0
  60. package/dist/enrichment/types.js +2 -0
  61. package/dist/enrichment/types.js.map +1 -0
  62. package/dist/enrichment/workflow.d.ts +35 -0
  63. package/dist/enrichment/workflow.d.ts.map +1 -0
  64. package/dist/enrichment/workflow.js +305 -0
  65. package/dist/enrichment/workflow.js.map +1 -0
  66. package/dist/generator/builtin-templates.d.ts +2 -1
  67. package/dist/generator/builtin-templates.d.ts.map +1 -1
  68. package/dist/generator/builtin-templates.js +161 -2
  69. package/dist/generator/builtin-templates.js.map +1 -1
  70. package/dist/generator/package-renderer.d.ts +29 -0
  71. package/dist/generator/package-renderer.d.ts.map +1 -0
  72. package/dist/generator/package-renderer.js +123 -0
  73. package/dist/generator/package-renderer.js.map +1 -0
  74. package/dist/generator/render.d.ts +14 -63
  75. package/dist/generator/render.d.ts.map +1 -1
  76. package/dist/generator/render.js +30 -176
  77. package/dist/generator/render.js.map +1 -1
  78. package/dist/index.js +42 -2
  79. package/dist/index.js.map +1 -1
  80. package/dist/licensing/installer.d.ts +51 -0
  81. package/dist/licensing/installer.d.ts.map +1 -0
  82. package/dist/licensing/installer.js +112 -0
  83. package/dist/licensing/installer.js.map +1 -0
  84. package/dist/licensing/license-manager.d.ts +6 -0
  85. package/dist/licensing/license-manager.d.ts.map +1 -1
  86. package/dist/licensing/license-manager.js +12 -35
  87. package/dist/licensing/license-manager.js.map +1 -1
  88. package/dist/licensing/template-fetch.d.ts +3 -2
  89. package/dist/licensing/template-fetch.d.ts.map +1 -1
  90. package/dist/licensing/template-fetch.js +10 -41
  91. package/dist/licensing/template-fetch.js.map +1 -1
  92. package/dist/scan/__tests__/context-gatherer.test.d.ts +2 -0
  93. package/dist/scan/__tests__/context-gatherer.test.d.ts.map +1 -0
  94. package/dist/scan/__tests__/context-gatherer.test.js +111 -0
  95. package/dist/scan/__tests__/context-gatherer.test.js.map +1 -0
  96. package/dist/scan/__tests__/merge.test.d.ts +2 -0
  97. package/dist/scan/__tests__/merge.test.d.ts.map +1 -0
  98. package/dist/scan/__tests__/merge.test.js +163 -0
  99. package/dist/scan/__tests__/merge.test.js.map +1 -0
  100. package/dist/scan/auto-bootstrap.d.ts +3 -0
  101. package/dist/scan/auto-bootstrap.d.ts.map +1 -0
  102. package/dist/scan/auto-bootstrap.js +30 -0
  103. package/dist/scan/auto-bootstrap.js.map +1 -0
  104. package/dist/scan/config.d.ts +24 -0
  105. package/dist/scan/config.d.ts.map +1 -0
  106. package/dist/scan/config.js +77 -0
  107. package/dist/scan/config.js.map +1 -0
  108. package/dist/scan/context/gatherer.d.ts +13 -0
  109. package/dist/scan/context/gatherer.d.ts.map +1 -0
  110. package/dist/scan/context/gatherer.js +97 -0
  111. package/dist/scan/context/gatherer.js.map +1 -0
  112. package/dist/scan/context/gitignore.d.ts +9 -0
  113. package/dist/scan/context/gitignore.d.ts.map +1 -0
  114. package/dist/scan/context/gitignore.js +50 -0
  115. package/dist/scan/context/gitignore.js.map +1 -0
  116. package/dist/scan/context/patterns.d.ts +14 -0
  117. package/dist/scan/context/patterns.d.ts.map +1 -0
  118. package/dist/scan/context/patterns.js +159 -0
  119. package/dist/scan/context/patterns.js.map +1 -0
  120. package/dist/scan/customs/writer.d.ts +6 -0
  121. package/dist/scan/customs/writer.d.ts.map +1 -0
  122. package/dist/scan/customs/writer.js +149 -0
  123. package/dist/scan/customs/writer.js.map +1 -0
  124. package/dist/scan/llm/anthropic.d.ts +11 -0
  125. package/dist/scan/llm/anthropic.d.ts.map +1 -0
  126. package/dist/scan/llm/anthropic.js +89 -0
  127. package/dist/scan/llm/anthropic.js.map +1 -0
  128. package/dist/scan/llm/factory.d.ts +4 -0
  129. package/dist/scan/llm/factory.d.ts.map +1 -0
  130. package/dist/scan/llm/factory.js +20 -0
  131. package/dist/scan/llm/factory.js.map +1 -0
  132. package/dist/scan/llm/ollama.d.ts +11 -0
  133. package/dist/scan/llm/ollama.d.ts.map +1 -0
  134. package/dist/scan/llm/ollama.js +64 -0
  135. package/dist/scan/llm/ollama.js.map +1 -0
  136. package/dist/scan/llm/openai.d.ts +12 -0
  137. package/dist/scan/llm/openai.d.ts.map +1 -0
  138. package/dist/scan/llm/openai.js +78 -0
  139. package/dist/scan/llm/openai.js.map +1 -0
  140. package/dist/scan/llm/types.d.ts +23 -0
  141. package/dist/scan/llm/types.d.ts.map +1 -0
  142. package/dist/scan/llm/types.js +3 -0
  143. package/dist/scan/llm/types.js.map +1 -0
  144. package/dist/scan/merge/diff-display.d.ts +11 -0
  145. package/dist/scan/merge/diff-display.d.ts.map +1 -0
  146. package/dist/scan/merge/diff-display.js +72 -0
  147. package/dist/scan/merge/diff-display.js.map +1 -0
  148. package/dist/scan/prompt/agent.d.ts +23 -0
  149. package/dist/scan/prompt/agent.d.ts.map +1 -0
  150. package/dist/scan/prompt/agent.js +95 -0
  151. package/dist/scan/prompt/agent.js.map +1 -0
  152. package/dist/scan/prompt/builder.d.ts +16 -0
  153. package/dist/scan/prompt/builder.d.ts.map +1 -0
  154. package/dist/scan/prompt/builder.js +64 -0
  155. package/dist/scan/prompt/builder.js.map +1 -0
  156. package/dist/scan/prompt/schema.d.ts +7 -0
  157. package/dist/scan/prompt/schema.d.ts.map +1 -0
  158. package/dist/scan/prompt/schema.js +52 -0
  159. package/dist/scan/prompt/schema.js.map +1 -0
  160. package/dist/security/scanner.d.ts +3 -40
  161. package/dist/security/scanner.d.ts.map +1 -1
  162. package/dist/security/scanner.js +3 -169
  163. package/dist/security/scanner.js.map +1 -1
  164. package/dist/utils/fetch.d.ts +9 -0
  165. package/dist/utils/fetch.d.ts.map +1 -0
  166. package/dist/utils/fetch.js +18 -0
  167. package/dist/utils/fetch.js.map +1 -0
  168. package/package.json +15 -12
@@ -0,0 +1,159 @@
1
+ // @nepopsx/cli — Framework-aware tiered file patterns for context gathering
2
+ const BASE_PATTERNS = {
3
+ tier1: [
4
+ 'README.md',
5
+ 'README.txt',
6
+ '.env.example',
7
+ 'docker-compose.yml',
8
+ 'docker-compose.yaml',
9
+ 'Dockerfile',
10
+ ],
11
+ tier2: [
12
+ 'ARCHITECTURE.md',
13
+ 'docs/**/*.md',
14
+ ],
15
+ tier3: [],
16
+ };
17
+ const FRAMEWORK_PATTERNS = {
18
+ nestjs: {
19
+ tier1: [
20
+ 'package.json',
21
+ 'tsconfig.json',
22
+ 'nest-cli.json',
23
+ 'src/app.module.ts',
24
+ 'src/main.ts',
25
+ 'src/**/*.module.ts',
26
+ ],
27
+ tier2: [
28
+ 'src/**/*.controller.ts',
29
+ 'src/**/*.service.ts',
30
+ 'src/**/*.entity.ts',
31
+ 'src/**/*.dto.ts',
32
+ 'src/database/**/*.ts',
33
+ ],
34
+ tier3: [
35
+ 'src/**/*.guard.ts',
36
+ 'src/**/*.decorator.ts',
37
+ 'src/**/*.pipe.ts',
38
+ ],
39
+ },
40
+ nextjs: {
41
+ tier1: [
42
+ 'package.json',
43
+ 'tsconfig.json',
44
+ 'next.config.ts',
45
+ 'next.config.js',
46
+ 'next.config.mjs',
47
+ 'src/app/layout.tsx',
48
+ 'src/app/page.tsx',
49
+ 'app/layout.tsx',
50
+ 'app/page.tsx',
51
+ ],
52
+ tier2: [
53
+ 'src/app/**/page.tsx',
54
+ 'src/app/**/layout.tsx',
55
+ 'src/components/**/*.tsx',
56
+ 'src/lib/**/*.ts',
57
+ 'src/types/**/*.ts',
58
+ ],
59
+ tier3: [
60
+ 'src/hooks/**/*.ts',
61
+ 'src/stores/**/*.ts',
62
+ 'src/actions/**/*.ts',
63
+ ],
64
+ },
65
+ react: {
66
+ tier1: [
67
+ 'package.json',
68
+ 'tsconfig.json',
69
+ 'vite.config.ts',
70
+ 'vite.config.js',
71
+ 'src/App.tsx',
72
+ 'src/App.jsx',
73
+ 'src/main.tsx',
74
+ 'src/index.tsx',
75
+ ],
76
+ tier2: [
77
+ 'src/components/**/*.tsx',
78
+ 'src/pages/**/*.tsx',
79
+ ],
80
+ tier3: [
81
+ 'src/hooks/**/*.ts',
82
+ 'src/store/**/*.ts',
83
+ 'src/api/**/*.ts',
84
+ ],
85
+ },
86
+ express: {
87
+ tier1: [
88
+ 'package.json',
89
+ 'tsconfig.json',
90
+ 'src/app.ts',
91
+ 'src/index.ts',
92
+ 'src/server.ts',
93
+ 'app.js',
94
+ 'index.js',
95
+ 'server.js',
96
+ ],
97
+ tier2: [
98
+ 'src/routes/**/*.ts',
99
+ 'src/controllers/**/*.ts',
100
+ 'src/middleware/**/*.ts',
101
+ 'src/models/**/*.ts',
102
+ ],
103
+ tier3: [
104
+ 'src/services/**/*.ts',
105
+ 'src/utils/**/*.ts',
106
+ ],
107
+ },
108
+ go: {
109
+ tier1: [
110
+ 'go.mod',
111
+ 'go.sum',
112
+ 'Makefile',
113
+ 'main.go',
114
+ 'cmd/*/main.go',
115
+ 'cmd/**/*.go',
116
+ ],
117
+ tier2: [
118
+ 'internal/**/*.go',
119
+ 'pkg/**/*.go',
120
+ 'api/**/*.go',
121
+ ],
122
+ tier3: [
123
+ '**/handler*.go',
124
+ '**/service*.go',
125
+ '**/store*.go',
126
+ ],
127
+ },
128
+ python: {
129
+ tier1: [
130
+ 'requirements.txt',
131
+ 'pyproject.toml',
132
+ 'setup.py',
133
+ 'setup.cfg',
134
+ 'Pipfile',
135
+ 'main.py',
136
+ 'app.py',
137
+ 'src/**/__init__.py',
138
+ 'src/**/main.py',
139
+ ],
140
+ tier2: [
141
+ 'src/**/*.py',
142
+ ],
143
+ tier3: [],
144
+ },
145
+ };
146
+ /**
147
+ * Get tiered file patterns for a given framework.
148
+ * The returned patterns are merged with the base patterns.
149
+ */
150
+ export function getPatternsForFramework(framework) {
151
+ const fw = framework?.toLowerCase();
152
+ const specific = fw ? (FRAMEWORK_PATTERNS[fw] ?? null) : null;
153
+ return {
154
+ tier1: [...BASE_PATTERNS.tier1, ...(specific?.tier1 ?? [])],
155
+ tier2: [...BASE_PATTERNS.tier2, ...(specific?.tier2 ?? [])],
156
+ tier3: [...BASE_PATTERNS.tier3, ...(specific?.tier3 ?? [])],
157
+ };
158
+ }
159
+ //# sourceMappingURL=patterns.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"patterns.js","sourceRoot":"","sources":["../../../src/scan/context/patterns.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import type { ScanResponse } from '@nepopsx/core';
2
+ /**
3
+ * Write or update customs files with scan-managed sections.
4
+ */
5
+ export declare function writeCustomsFiles(customsDir: string, scanResults: ScanResponse, dryRun: boolean): void;
6
+ //# sourceMappingURL=writer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"writer.d.ts","sourceRoot":"","sources":["../../../src/scan/customs/writer.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,eAAe,CAAC;AAuIlD;;GAEG;AACH,wBAAgB,iBAAiB,CAC/B,UAAU,EAAE,MAAM,EAClB,WAAW,EAAE,YAAY,EACzB,MAAM,EAAE,OAAO,GACd,IAAI,CAsBN"}
@@ -0,0 +1,149 @@
1
+ // @nepopsx/cli — Customs file writer with marker-based merge
2
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
3
+ import { join } from 'node:path';
4
+ const MARKER_START = '<!-- nepopsx-scan-managed:start -->';
5
+ const MARKER_END = '<!-- nepopsx-scan-managed:end -->';
6
+ const FILE_HEADER = `<!-- This file is partially managed by \`nepopsx scan\`.
7
+ Sections between ${MARKER_START} and ${MARKER_END} are auto-generated.
8
+ Content outside those markers is preserved across re-runs. -->
9
+
10
+ `;
11
+ /**
12
+ * Merge scan-managed content into an existing file while preserving user sections.
13
+ * If the file does not exist, create it with the managed block.
14
+ */
15
+ function mergeCustomsFile(filePath, managedContent, dryRun) {
16
+ const newBlock = `${MARKER_START}\n${managedContent.trim()}\n${MARKER_END}`;
17
+ if (!existsSync(filePath)) {
18
+ if (!dryRun) {
19
+ writeFileSync(filePath, FILE_HEADER + newBlock + '\n', 'utf-8');
20
+ }
21
+ return true;
22
+ }
23
+ const existing = readFileSync(filePath, 'utf-8');
24
+ const startIdx = existing.indexOf(MARKER_START);
25
+ const endIdx = existing.indexOf(MARKER_END);
26
+ let updated;
27
+ if (startIdx === -1 || endIdx === -1) {
28
+ // No markers — append managed block
29
+ updated = existing.trimEnd() + '\n\n' + newBlock + '\n';
30
+ }
31
+ else {
32
+ // Replace between markers
33
+ const before = existing.slice(0, startIdx);
34
+ const after = existing.slice(endIdx + MARKER_END.length);
35
+ updated = before + newBlock + after;
36
+ }
37
+ if (updated === existing)
38
+ return false;
39
+ if (!dryRun) {
40
+ writeFileSync(filePath, updated, 'utf-8');
41
+ }
42
+ return true;
43
+ }
44
+ /**
45
+ * Build the content for architecture-patterns.md from scan results.
46
+ */
47
+ function buildArchitecturePatterns(scan) {
48
+ const lines = ['## Architecture Patterns', ''];
49
+ for (const svc of scan.services) {
50
+ if (!svc.patterns || svc.patterns.length === 0)
51
+ continue;
52
+ lines.push(`### ${svc.service_name}`);
53
+ for (const pattern of svc.patterns) {
54
+ lines.push(`- ${pattern}`);
55
+ }
56
+ lines.push('');
57
+ }
58
+ return lines.join('\n');
59
+ }
60
+ /**
61
+ * Build the content for coding-conventions.md from scan results.
62
+ */
63
+ function buildCodingConventions(scan) {
64
+ const lines = ['## Coding Conventions', ''];
65
+ for (const svc of scan.services) {
66
+ if (!svc.conventions || svc.conventions.length === 0)
67
+ continue;
68
+ lines.push(`### ${svc.service_name}`);
69
+ for (const conv of svc.conventions) {
70
+ lines.push(`- ${conv}`);
71
+ }
72
+ lines.push('');
73
+ }
74
+ if (scan.cross_service?.conventions) {
75
+ lines.push('### Workspace-wide');
76
+ const c = scan.cross_service.conventions;
77
+ if (c.commits)
78
+ lines.push(`- Commit style: ${c.commits}`);
79
+ if (c.branching)
80
+ lines.push(`- Branching: ${c.branching}`);
81
+ if (c.pr_required !== undefined)
82
+ lines.push(`- PR required: ${c.pr_required}`);
83
+ lines.push('');
84
+ }
85
+ return lines.join('\n');
86
+ }
87
+ /**
88
+ * Build the content for service-communication.md from scan results.
89
+ */
90
+ function buildServiceCommunication(scan) {
91
+ const lines = ['## Service Communication', ''];
92
+ if (scan.cross_service?.service_communication) {
93
+ for (const comm of scan.cross_service.service_communication) {
94
+ lines.push(`- **${comm.from}** → **${comm.to}** (${comm.protocol}): ${comm.description}`);
95
+ }
96
+ lines.push('');
97
+ }
98
+ for (const svc of scan.services) {
99
+ if (!svc.communication || svc.communication.length === 0)
100
+ continue;
101
+ lines.push(`### ${svc.service_name} outbound`);
102
+ for (const c of svc.communication) {
103
+ lines.push(`- → **${c.target}** via ${c.protocol}: ${c.description}`);
104
+ }
105
+ lines.push('');
106
+ }
107
+ return lines.join('\n');
108
+ }
109
+ /**
110
+ * Build the content for gotchas.md from scan results.
111
+ */
112
+ function buildGotchas(scan) {
113
+ const lines = ['## Known Gotchas', ''];
114
+ for (const svc of scan.services) {
115
+ if (!svc.gotchas || svc.gotchas.length === 0)
116
+ continue;
117
+ lines.push(`### ${svc.service_name}`);
118
+ for (const g of svc.gotchas) {
119
+ lines.push(`- ${g}`);
120
+ }
121
+ lines.push('');
122
+ }
123
+ return lines.join('\n');
124
+ }
125
+ /**
126
+ * Write or update customs files with scan-managed sections.
127
+ */
128
+ export function writeCustomsFiles(customsDir, scanResults, dryRun) {
129
+ if (!dryRun && !existsSync(customsDir)) {
130
+ mkdirSync(customsDir, { recursive: true });
131
+ }
132
+ const files = [
133
+ { name: 'architecture-patterns.md', content: buildArchitecturePatterns(scanResults) },
134
+ { name: 'coding-conventions.md', content: buildCodingConventions(scanResults) },
135
+ { name: 'service-communication.md', content: buildServiceCommunication(scanResults) },
136
+ { name: 'gotchas.md', content: buildGotchas(scanResults) },
137
+ ];
138
+ for (const { name, content } of files) {
139
+ if (content.trim().split('\n').length <= 2)
140
+ continue; // Skip empty sections
141
+ const filePath = join(customsDir, name);
142
+ const changed = mergeCustomsFile(filePath, content, dryRun);
143
+ const prefix = dryRun ? '[dry-run] ' : '';
144
+ if (changed) {
145
+ console.log(` ${prefix}${existsSync(filePath) && !dryRun ? 'updated' : 'created'}: customs/${name}`);
146
+ }
147
+ }
148
+ }
149
+ //# sourceMappingURL=writer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"writer.js","sourceRoot":"","sources":["../../../src/scan/customs/writer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { LlmProvider, LlmRequest, LlmResponse } from './types.js';
2
+ export declare class AnthropicProvider implements LlmProvider {
3
+ private readonly apiKey;
4
+ private readonly model;
5
+ readonly name: string;
6
+ readonly supportsJsonMode = true;
7
+ constructor(apiKey: string, model?: string);
8
+ estimateTokens(text: string): number;
9
+ chat(request: LlmRequest): Promise<LlmResponse>;
10
+ }
11
+ //# sourceMappingURL=anthropic.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"anthropic.d.ts","sourceRoot":"","sources":["../../../src/scan/llm/anthropic.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AASvE,qBAAa,iBAAkB,YAAW,WAAW;IAKjD,OAAO,CAAC,QAAQ,CAAC,MAAM;IACvB,OAAO,CAAC,QAAQ,CAAC,KAAK;IALxB,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,gBAAgB,QAAQ;gBAGd,MAAM,EAAE,MAAM,EACd,KAAK,GAAE,MAAsB;IAKhD,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAI9B,IAAI,CAAC,OAAO,EAAE,UAAU,GAAG,OAAO,CAAC,WAAW,CAAC;CAkFtD"}
@@ -0,0 +1,89 @@
1
+ import { fetchWithTimeout } from '../../utils/fetch.js';
2
+ const DEFAULT_MODEL = 'claude-3-5-sonnet-20241022';
3
+ const ENDPOINT = 'https://api.anthropic.com/v1/messages';
4
+ const TIMEOUT_MS = 120_000;
5
+ const MAX_RETRIES = 3;
6
+ const ANTHROPIC_VERSION = '2023-06-01';
7
+ export class AnthropicProvider {
8
+ apiKey;
9
+ model;
10
+ name;
11
+ supportsJsonMode = true;
12
+ constructor(apiKey, model = DEFAULT_MODEL) {
13
+ this.apiKey = apiKey;
14
+ this.model = model;
15
+ this.name = `anthropic:${model}`;
16
+ }
17
+ estimateTokens(text) {
18
+ return Math.ceil(text.length / 3.5);
19
+ }
20
+ async chat(request) {
21
+ // Use tool_use for structured output when a schema is provided
22
+ const tools = request.jsonSchema
23
+ ? [
24
+ {
25
+ name: 'scan_response',
26
+ description: 'Return the structured scan analysis result',
27
+ input_schema: request.jsonSchema,
28
+ },
29
+ ]
30
+ : undefined;
31
+ const body = {
32
+ model: this.model,
33
+ max_tokens: request.maxTokens,
34
+ system: request.systemPrompt,
35
+ messages: [{ role: 'user', content: request.userPrompt }],
36
+ };
37
+ if (tools) {
38
+ body.tools = tools;
39
+ body.tool_choice = { type: 'tool', name: 'scan_response' };
40
+ }
41
+ let lastError = null;
42
+ for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
43
+ if (attempt > 0) {
44
+ await new Promise((r) => setTimeout(r, 1000 * Math.pow(2, attempt - 1)));
45
+ }
46
+ try {
47
+ const res = await fetchWithTimeout(ENDPOINT, {
48
+ method: 'POST',
49
+ headers: {
50
+ 'Content-Type': 'application/json',
51
+ 'x-api-key': this.apiKey,
52
+ 'anthropic-version': ANTHROPIC_VERSION,
53
+ },
54
+ body: JSON.stringify(body),
55
+ }, TIMEOUT_MS);
56
+ if (!res.ok) {
57
+ const text = await res.text();
58
+ throw new Error(`Anthropic API error ${res.status}: ${text.slice(0, 200)}`);
59
+ }
60
+ const data = (await res.json());
61
+ // Extract content: tool_use returns input as JSON; text blocks return raw text
62
+ let content = '';
63
+ for (const block of data.content) {
64
+ if (block.type === 'tool_use' && block.input !== undefined) {
65
+ content = JSON.stringify(block.input);
66
+ break;
67
+ }
68
+ if (block.type === 'text' && block.text) {
69
+ content = block.text;
70
+ }
71
+ }
72
+ return {
73
+ content,
74
+ inputTokens: data.usage?.input_tokens ?? 0,
75
+ outputTokens: data.usage?.output_tokens ?? 0,
76
+ model: data.model,
77
+ };
78
+ }
79
+ catch (err) {
80
+ lastError = err instanceof Error ? err : new Error(String(err));
81
+ if (lastError.name === 'AbortError') {
82
+ throw new Error(`Anthropic request timed out after ${TIMEOUT_MS}ms`);
83
+ }
84
+ }
85
+ }
86
+ throw lastError ?? new Error('Anthropic request failed after retries');
87
+ }
88
+ }
89
+ //# sourceMappingURL=anthropic.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"anthropic.js","sourceRoot":"","sources":["../../../src/scan/llm/anthropic.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AAExD,MAAM,aAAa,GAAG,4BAA4B,CAAC;AACnD,MAAM,QAAQ,GAAG,uCAAuC,CAAC;AACzD,MAAM,UAAU,GAAG,OAAO,CAAC;AAC3B,MAAM,WAAW,GAAG,CAAC,CAAC;AACtB,MAAM,iBAAiB,GAAG,YAAY,CAAC;AAEvC,MAAM,OAAO,iBAAiB;IAKT;IACA;IALV,IAAI,CAAS;IACb,gBAAgB,GAAG,IAAI,CAAC;IAEjC,YACmB,MAAc,EACd,QAAgB,aAAa;QAD7B,WAAM,GAAN,MAAM,CAAQ;QACd,UAAK,GAAL,KAAK,CAAwB;QAE9C,IAAI,CAAC,IAAI,GAAG,aAAa,KAAK,EAAE,CAAC;IACnC,CAAC;IAED,cAAc,CAAC,IAAY;QACzB,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,GAAG,CAAC,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,OAAmB;QAC5B,+DAA+D;QAC/D,MAAM,KAAK,GAAG,OAAO,CAAC,UAAU;YAC9B,CAAC,CAAC;gBACE;oBACE,IAAI,EAAE,eAAe;oBACrB,WAAW,EAAE,4CAA4C;oBACzD,YAAY,EAAE,OAAO,CAAC,UAAU;iBACjC;aACF;YACH,CAAC,CAAC,SAAS,CAAC;QAEd,MAAM,IAAI,GAA4B;YACpC,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,UAAU,EAAE,OAAO,CAAC,SAAS;YAC7B,MAAM,EAAE,OAAO,CAAC,YAAY;YAC5B,QAAQ,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,UAAU,EAAE,CAAC;SAC1D,CAAC;QAEF,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;YACnB,IAAI,CAAC,WAAW,GAAG,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,eAAe,EAAE,CAAC;QAC7D,CAAC;QAED,IAAI,SAAS,GAAiB,IAAI,CAAC;QACnC,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,GAAG,WAAW,EAAE,OAAO,EAAE,EAAE,CAAC;YACvD,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;gBAChB,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;YAC3E,CAAC;YACD,IAAI,CAAC;gBACH,MAAM,GAAG,GAAG,MAAM,gBAAgB,CAChC,QAAQ,EACR;oBACE,MAAM,EAAE,MAAM;oBACd,OAAO,EAAE;wBACP,cAAc,EAAE,kBAAkB;wBAClC,WAAW,EAAE,IAAI,CAAC,MAAM;wBACxB,mBAAmB,EAAE,iBAAiB;qBACvC;oBACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;iBAC3B,EACD,UAAU,CACX,CAAC;gBAEF,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;oBACZ,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC;oBAC9B,MAAM,IAAI,KAAK,CAAC,uBAAuB,GAAG,CAAC,MAAM,KAAK,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC;gBAC9E,CAAC;gBAED,MAAM,IAAI,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,CAI7B,CAAC;gBAEF,+EAA+E;gBAC/E,IAAI,OAAO,GAAG,EAAE,CAAC;gBACjB,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;oBACjC,IAAI,KAAK,CAAC,IAAI,KAAK,UAAU,IAAI,KAAK,CAAC,KAAK,KAAK,SAAS,EAAE,CAAC;wBAC3D,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;wBACtC,MAAM;oBACR,CAAC;oBACD,IAAI,KAAK,CAAC,IAAI,KAAK,MAAM,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC;wBACxC,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC;oBACvB,CAAC;gBACH,CAAC;gBAED,OAAO;oBACL,OAAO;oBACP,WAAW,EAAE,IAAI,CAAC,KAAK,EAAE,YAAY,IAAI,CAAC;oBAC1C,YAAY,EAAE,IAAI,CAAC,KAAK,EAAE,aAAa,IAAI,CAAC;oBAC5C,KAAK,EAAE,IAAI,CAAC,KAAK;iBAClB,CAAC;YACJ,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,SAAS,GAAG,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;gBAChE,IAAI,SAAS,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;oBACpC,MAAM,IAAI,KAAK,CAAC,qCAAqC,UAAU,IAAI,CAAC,CAAC;gBACvE,CAAC;YACH,CAAC;QACH,CAAC;QACD,MAAM,SAAS,IAAI,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;IACzE,CAAC;CACF"}
@@ -0,0 +1,4 @@
1
+ import type { LlmProvider } from './types.js';
2
+ import type { ResolvedScanConfig } from '../config.js';
3
+ export declare function createProvider(config: ResolvedScanConfig): LlmProvider;
4
+ //# sourceMappingURL=factory.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"factory.d.ts","sourceRoot":"","sources":["../../../src/scan/llm/factory.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAI9C,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAEvD,wBAAgB,cAAc,CAAC,MAAM,EAAE,kBAAkB,GAAG,WAAW,CAsBtE"}
@@ -0,0 +1,20 @@
1
+ import { OpenAiProvider } from './openai.js';
2
+ import { AnthropicProvider } from './anthropic.js';
3
+ import { OllamaProvider } from './ollama.js';
4
+ export function createProvider(config) {
5
+ switch (config.provider) {
6
+ case 'openai':
7
+ return new OpenAiProvider(config.apiKey ?? '', config.model);
8
+ case 'compatible':
9
+ return new OpenAiProvider(config.apiKey ?? '', config.model, config.endpoint ?? 'https://api.openai.com/v1');
10
+ case 'anthropic':
11
+ return new AnthropicProvider(config.apiKey ?? '', config.model);
12
+ case 'ollama':
13
+ return new OllamaProvider(config.endpoint ?? 'http://localhost:11434', config.model);
14
+ default: {
15
+ const exhaustive = config.provider;
16
+ throw new Error(`Unknown provider: ${String(exhaustive)}`);
17
+ }
18
+ }
19
+ }
20
+ //# sourceMappingURL=factory.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"factory.js","sourceRoot":"","sources":["../../../src/scan/llm/factory.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAG7C,MAAM,UAAU,cAAc,CAAC,MAA0B;IACvD,QAAQ,MAAM,CAAC,QAAQ,EAAE,CAAC;QACxB,KAAK,QAAQ;YACX,OAAO,IAAI,cAAc,CAAC,MAAM,CAAC,MAAM,IAAI,EAAE,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;QAC/D,KAAK,YAAY;YACf,OAAO,IAAI,cAAc,CACvB,MAAM,CAAC,MAAM,IAAI,EAAE,EACnB,MAAM,CAAC,KAAK,EACZ,MAAM,CAAC,QAAQ,IAAI,2BAA2B,CAC/C,CAAC;QACJ,KAAK,WAAW;YACd,OAAO,IAAI,iBAAiB,CAAC,MAAM,CAAC,MAAM,IAAI,EAAE,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;QAClE,KAAK,QAAQ;YACX,OAAO,IAAI,cAAc,CACvB,MAAM,CAAC,QAAQ,IAAI,wBAAwB,EAC3C,MAAM,CAAC,KAAK,CACb,CAAC;QACJ,OAAO,CAAC,CAAC,CAAC;YACR,MAAM,UAAU,GAAU,MAAM,CAAC,QAAQ,CAAC;YAC1C,MAAM,IAAI,KAAK,CAAC,qBAAqB,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;QAC7D,CAAC;IACH,CAAC;AACH,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { LlmProvider, LlmRequest, LlmResponse } from './types.js';
2
+ export declare class OllamaProvider implements LlmProvider {
3
+ private readonly baseUrl;
4
+ private readonly model;
5
+ readonly name: string;
6
+ readonly supportsJsonMode = false;
7
+ constructor(baseUrl?: string, model?: string);
8
+ estimateTokens(text: string): number;
9
+ chat(request: LlmRequest): Promise<LlmResponse>;
10
+ }
11
+ //# sourceMappingURL=ollama.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ollama.d.ts","sourceRoot":"","sources":["../../../src/scan/llm/ollama.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAQvE,qBAAa,cAAe,YAAW,WAAW;IAK9C,OAAO,CAAC,QAAQ,CAAC,OAAO;IACxB,OAAO,CAAC,QAAQ,CAAC,KAAK;IALxB,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,gBAAgB,SAAS;gBAGf,OAAO,GAAE,MAAyB,EAClC,KAAK,GAAE,MAAsB;IAKhD,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAI9B,IAAI,CAAC,OAAO,EAAE,UAAU,GAAG,OAAO,CAAC,WAAW,CAAC;CAyDtD"}
@@ -0,0 +1,64 @@
1
+ import { fetchWithTimeout } from '../../utils/fetch.js';
2
+ const DEFAULT_MODEL = 'llama3.2';
3
+ const DEFAULT_BASE_URL = 'http://localhost:11434';
4
+ const TIMEOUT_MS = 180_000; // Local models can be slower
5
+ const MAX_RETRIES = 5; // Higher retry budget for local models
6
+ export class OllamaProvider {
7
+ baseUrl;
8
+ model;
9
+ name;
10
+ supportsJsonMode = false; // Unconstrained JSON — needs strict validation
11
+ constructor(baseUrl = DEFAULT_BASE_URL, model = DEFAULT_MODEL) {
12
+ this.baseUrl = baseUrl;
13
+ this.model = model;
14
+ this.name = `ollama:${model}`;
15
+ }
16
+ estimateTokens(text) {
17
+ return Math.ceil(text.length / 3.5);
18
+ }
19
+ async chat(request) {
20
+ const url = `${this.baseUrl.replace(/\/$/, '')}/api/chat`;
21
+ const body = {
22
+ model: this.model,
23
+ stream: false,
24
+ format: 'json',
25
+ options: { temperature: request.temperature },
26
+ messages: [
27
+ { role: 'system', content: request.systemPrompt },
28
+ { role: 'user', content: request.userPrompt },
29
+ ],
30
+ };
31
+ let lastError = null;
32
+ for (let attempt = 0; attempt < MAX_RETRIES; attempt++) {
33
+ if (attempt > 0) {
34
+ await new Promise((r) => setTimeout(r, 1000 * Math.pow(2, Math.min(attempt - 1, 4))));
35
+ }
36
+ try {
37
+ const res = await fetchWithTimeout(url, {
38
+ method: 'POST',
39
+ headers: { 'Content-Type': 'application/json' },
40
+ body: JSON.stringify(body),
41
+ }, TIMEOUT_MS);
42
+ if (!res.ok) {
43
+ const text = await res.text();
44
+ throw new Error(`Ollama error ${res.status}: ${text.slice(0, 200)}`);
45
+ }
46
+ const data = (await res.json());
47
+ return {
48
+ content: data.message?.content ?? '',
49
+ inputTokens: data.prompt_eval_count ?? 0,
50
+ outputTokens: data.eval_count ?? 0,
51
+ model: data.model,
52
+ };
53
+ }
54
+ catch (err) {
55
+ lastError = err instanceof Error ? err : new Error(String(err));
56
+ if (lastError.name === 'AbortError') {
57
+ throw new Error(`Ollama request timed out after ${TIMEOUT_MS}ms`);
58
+ }
59
+ }
60
+ }
61
+ throw lastError ?? new Error('Ollama request failed after retries');
62
+ }
63
+ }
64
+ //# sourceMappingURL=ollama.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ollama.js","sourceRoot":"","sources":["../../../src/scan/llm/ollama.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ import type { LlmProvider, LlmRequest, LlmResponse } from './types.js';
2
+ export declare class OpenAiProvider implements LlmProvider {
3
+ private readonly apiKey;
4
+ private readonly model;
5
+ private readonly endpoint;
6
+ readonly name: string;
7
+ readonly supportsJsonMode = true;
8
+ constructor(apiKey: string, model?: string, endpoint?: string);
9
+ estimateTokens(text: string): number;
10
+ chat(request: LlmRequest): Promise<LlmResponse>;
11
+ }
12
+ //# sourceMappingURL=openai.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"openai.d.ts","sourceRoot":"","sources":["../../../src/scan/llm/openai.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,WAAW,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAQvE,qBAAa,cAAe,YAAW,WAAW;IAK9C,OAAO,CAAC,QAAQ,CAAC,MAAM;IACvB,OAAO,CAAC,QAAQ,CAAC,KAAK;IACtB,OAAO,CAAC,QAAQ,CAAC,QAAQ;IAN3B,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,gBAAgB,QAAQ;gBAGd,MAAM,EAAE,MAAM,EACd,KAAK,GAAE,MAAsB,EAC7B,QAAQ,GAAE,MAAyB;IAKtD,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAI9B,IAAI,CAAC,OAAO,EAAE,UAAU,GAAG,OAAO,CAAC,WAAW,CAAC;CAqEtD"}