@intrect/openswarm 0.2.2 → 0.4.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 (176) hide show
  1. package/README.md +236 -331
  2. package/config.example.yaml +36 -13
  3. package/dist/adapters/agenticLoop.d.ts +90 -0
  4. package/dist/adapters/agenticLoop.d.ts.map +1 -0
  5. package/dist/adapters/agenticLoop.js +141 -0
  6. package/dist/adapters/agenticLoop.js.map +1 -0
  7. package/dist/adapters/base.d.ts.map +1 -1
  8. package/dist/adapters/base.js +4 -0
  9. package/dist/adapters/base.js.map +1 -1
  10. package/dist/adapters/cryptoQuantAdapter.js +1 -1
  11. package/dist/adapters/cryptoQuantAdapter.js.map +1 -1
  12. package/dist/adapters/gpt.d.ts +19 -0
  13. package/dist/adapters/gpt.d.ts.map +1 -0
  14. package/dist/adapters/gpt.js +251 -0
  15. package/dist/adapters/gpt.js.map +1 -0
  16. package/dist/adapters/index.d.ts +2 -0
  17. package/dist/adapters/index.d.ts.map +1 -1
  18. package/dist/adapters/index.js +6 -0
  19. package/dist/adapters/index.js.map +1 -1
  20. package/dist/adapters/local.d.ts +31 -0
  21. package/dist/adapters/local.d.ts.map +1 -0
  22. package/dist/adapters/local.js +320 -0
  23. package/dist/adapters/local.js.map +1 -0
  24. package/dist/adapters/tools.d.ts +30 -0
  25. package/dist/adapters/tools.d.ts.map +1 -0
  26. package/dist/adapters/tools.js +219 -0
  27. package/dist/adapters/tools.js.map +1 -0
  28. package/dist/adapters/types.d.ts +6 -1
  29. package/dist/adapters/types.d.ts.map +1 -1
  30. package/dist/agents/pairPipeline.d.ts +7 -0
  31. package/dist/agents/pairPipeline.d.ts.map +1 -1
  32. package/dist/agents/pairPipeline.js +99 -7
  33. package/dist/agents/pairPipeline.js.map +1 -1
  34. package/dist/agents/pipelineGuards.d.ts.map +1 -1
  35. package/dist/agents/pipelineGuards.js +84 -2
  36. package/dist/agents/pipelineGuards.js.map +1 -1
  37. package/dist/agents/worker.d.ts +3 -0
  38. package/dist/agents/worker.d.ts.map +1 -1
  39. package/dist/agents/worker.js +1 -0
  40. package/dist/agents/worker.js.map +1 -1
  41. package/dist/auth/index.d.ts +3 -0
  42. package/dist/auth/index.d.ts.map +1 -0
  43. package/dist/auth/index.js +6 -0
  44. package/dist/auth/index.js.map +1 -0
  45. package/dist/auth/oauthPkce.d.ts +21 -0
  46. package/dist/auth/oauthPkce.d.ts.map +1 -0
  47. package/dist/auth/oauthPkce.js +212 -0
  48. package/dist/auth/oauthPkce.js.map +1 -0
  49. package/dist/auth/oauthStore.d.ts +24 -0
  50. package/dist/auth/oauthStore.d.ts.map +1 -0
  51. package/dist/auth/oauthStore.js +96 -0
  52. package/dist/auth/oauthStore.js.map +1 -0
  53. package/dist/automation/autonomousRunner.d.ts +5 -5
  54. package/dist/automation/runnerTypes.d.ts +1 -1
  55. package/dist/automation/runnerTypes.d.ts.map +1 -1
  56. package/dist/cli/authHandler.d.ts +16 -0
  57. package/dist/cli/authHandler.d.ts.map +1 -0
  58. package/dist/cli/authHandler.js +93 -0
  59. package/dist/cli/authHandler.js.map +1 -0
  60. package/dist/cli/checkHandler.d.ts +25 -0
  61. package/dist/cli/checkHandler.d.ts.map +1 -0
  62. package/dist/cli/checkHandler.js +465 -0
  63. package/dist/cli/checkHandler.js.map +1 -0
  64. package/dist/cli.js +64 -0
  65. package/dist/cli.js.map +1 -1
  66. package/dist/core/config.d.ts +17 -4
  67. package/dist/core/config.d.ts.map +1 -1
  68. package/dist/core/config.js +21 -8
  69. package/dist/core/config.js.map +1 -1
  70. package/dist/core/service.d.ts.map +1 -1
  71. package/dist/core/service.js +18 -8
  72. package/dist/core/service.js.map +1 -1
  73. package/dist/core/types.d.ts +4 -2
  74. package/dist/core/types.d.ts.map +1 -1
  75. package/dist/issues/graphql/resolvers.d.ts +252 -0
  76. package/dist/issues/graphql/resolvers.d.ts.map +1 -0
  77. package/dist/issues/graphql/resolvers.js +88 -0
  78. package/dist/issues/graphql/resolvers.js.map +1 -0
  79. package/dist/issues/graphql/server.d.ts +13 -0
  80. package/dist/issues/graphql/server.d.ts.map +1 -0
  81. package/dist/issues/graphql/server.js +56 -0
  82. package/dist/issues/graphql/server.js.map +1 -0
  83. package/dist/issues/graphql/typeDefs.d.ts +2 -0
  84. package/dist/issues/graphql/typeDefs.d.ts.map +1 -0
  85. package/dist/issues/graphql/typeDefs.js +251 -0
  86. package/dist/issues/graphql/typeDefs.js.map +1 -0
  87. package/dist/issues/index.d.ts +8 -0
  88. package/dist/issues/index.d.ts.map +1 -0
  89. package/dist/issues/index.js +11 -0
  90. package/dist/issues/index.js.map +1 -0
  91. package/dist/issues/issueBoardHtml.d.ts +2 -0
  92. package/dist/issues/issueBoardHtml.d.ts.map +1 -0
  93. package/dist/issues/issueBoardHtml.js +677 -0
  94. package/dist/issues/issueBoardHtml.js.map +1 -0
  95. package/dist/issues/linearBridge.d.ts +27 -0
  96. package/dist/issues/linearBridge.d.ts.map +1 -0
  97. package/dist/issues/linearBridge.js +211 -0
  98. package/dist/issues/linearBridge.js.map +1 -0
  99. package/dist/issues/memoryBridge.d.ts +35 -0
  100. package/dist/issues/memoryBridge.d.ts.map +1 -0
  101. package/dist/issues/memoryBridge.js +184 -0
  102. package/dist/issues/memoryBridge.js.map +1 -0
  103. package/dist/issues/schema.d.ts +162 -0
  104. package/dist/issues/schema.d.ts.map +1 -0
  105. package/dist/issues/schema.js +121 -0
  106. package/dist/issues/schema.js.map +1 -0
  107. package/dist/issues/sqliteStore.d.ts +90 -0
  108. package/dist/issues/sqliteStore.d.ts.map +1 -0
  109. package/dist/issues/sqliteStore.js +488 -0
  110. package/dist/issues/sqliteStore.js.map +1 -0
  111. package/dist/knowledge/index.d.ts.map +1 -1
  112. package/dist/knowledge/index.js +9 -3
  113. package/dist/knowledge/index.js.map +1 -1
  114. package/dist/linear/linear.d.ts +4 -0
  115. package/dist/linear/linear.d.ts.map +1 -1
  116. package/dist/linear/linear.js +27 -0
  117. package/dist/linear/linear.js.map +1 -1
  118. package/dist/locale/prompts/en.d.ts.map +1 -1
  119. package/dist/locale/prompts/en.js +32 -2
  120. package/dist/locale/prompts/en.js.map +1 -1
  121. package/dist/locale/prompts/ko.d.ts.map +1 -1
  122. package/dist/locale/prompts/ko.js +32 -2
  123. package/dist/locale/prompts/ko.js.map +1 -1
  124. package/dist/locale/types.d.ts +17 -0
  125. package/dist/locale/types.d.ts.map +1 -1
  126. package/dist/registry/bsDetector.d.ts +24 -0
  127. package/dist/registry/bsDetector.d.ts.map +1 -0
  128. package/dist/registry/bsDetector.js +276 -0
  129. package/dist/registry/bsDetector.js.map +1 -0
  130. package/dist/registry/entityScanner.d.ts +36 -0
  131. package/dist/registry/entityScanner.d.ts.map +1 -0
  132. package/dist/registry/entityScanner.js +693 -0
  133. package/dist/registry/entityScanner.js.map +1 -0
  134. package/dist/registry/graphql/resolvers.d.ts +778 -0
  135. package/dist/registry/graphql/resolvers.d.ts.map +1 -0
  136. package/dist/registry/graphql/resolvers.js +127 -0
  137. package/dist/registry/graphql/resolvers.js.map +1 -0
  138. package/dist/registry/graphql/typeDefs.d.ts +2 -0
  139. package/dist/registry/graphql/typeDefs.d.ts.map +1 -0
  140. package/dist/registry/graphql/typeDefs.js +276 -0
  141. package/dist/registry/graphql/typeDefs.js.map +1 -0
  142. package/dist/registry/index.d.ts +12 -0
  143. package/dist/registry/index.d.ts.map +1 -0
  144. package/dist/registry/index.js +18 -0
  145. package/dist/registry/index.js.map +1 -0
  146. package/dist/registry/issueBridge.d.ts +8 -0
  147. package/dist/registry/issueBridge.d.ts.map +1 -0
  148. package/dist/registry/issueBridge.js +30 -0
  149. package/dist/registry/issueBridge.js.map +1 -0
  150. package/dist/registry/memoryBridge.d.ts +13 -0
  151. package/dist/registry/memoryBridge.d.ts.map +1 -0
  152. package/dist/registry/memoryBridge.js +60 -0
  153. package/dist/registry/memoryBridge.js.map +1 -0
  154. package/dist/registry/schema.d.ts +307 -0
  155. package/dist/registry/schema.d.ts.map +1 -0
  156. package/dist/registry/schema.js +139 -0
  157. package/dist/registry/schema.js.map +1 -0
  158. package/dist/registry/sqliteStore.d.ts +101 -0
  159. package/dist/registry/sqliteStore.d.ts.map +1 -0
  160. package/dist/registry/sqliteStore.js +688 -0
  161. package/dist/registry/sqliteStore.js.map +1 -0
  162. package/dist/support/chatBackend.d.ts.map +1 -1
  163. package/dist/support/chatBackend.js +35 -4
  164. package/dist/support/chatBackend.js.map +1 -1
  165. package/dist/support/chatTui.d.ts.map +1 -1
  166. package/dist/support/chatTui.js +109 -3
  167. package/dist/support/chatTui.js.map +1 -1
  168. package/dist/support/dashboardHtml.d.ts +1 -1
  169. package/dist/support/dashboardHtml.d.ts.map +1 -1
  170. package/dist/support/dashboardHtml.js +1 -0
  171. package/dist/support/dashboardHtml.js.map +1 -1
  172. package/dist/support/web.d.ts.map +1 -1
  173. package/dist/support/web.js +16 -3
  174. package/dist/support/web.js.map +1 -1
  175. package/package.json +8 -2
  176. package/templates/TOOLS.md +2 -2
@@ -0,0 +1,276 @@
1
+ // ============================================
2
+ // OpenSwarm - BS Detector
3
+ // Created: 2026-04-10
4
+ // Purpose: 소스 코드에서 BS 패턴을 탐지하는 정적 분석 엔진
5
+ // ============================================
6
+ import { readFile } from 'node:fs/promises';
7
+ import { extname } from 'node:path';
8
+ // 테스트 파일 판별
9
+ function isTestPath(filePath) {
10
+ return /\.test\.|\.spec\.|_test\.|test_|__tests__/.test(filePath);
11
+ }
12
+ const BS_PATTERNS = [
13
+ // ============ CRITICAL ============
14
+ // 예외 은폐
15
+ {
16
+ severity: 'critical',
17
+ category: 'exception_hiding',
18
+ message: '빈 catch 블록 — 예외를 완전히 무시',
19
+ pattern: /catch\s*(?:\([^)]*\))?\s*\{\s*\}/,
20
+ excludeIf: (_line, fp) => isTestPath(fp) || /Html\.ts$/.test(fp),
21
+ },
22
+ {
23
+ severity: 'critical',
24
+ category: 'exception_hiding',
25
+ message: 'except: pass — Python 예외 은폐',
26
+ pattern: /except\s*:\s*pass/,
27
+ languages: ['python'],
28
+ },
29
+ {
30
+ severity: 'critical',
31
+ category: 'exception_hiding',
32
+ message: 'except Exception: pass — 모든 예외 무시',
33
+ pattern: /except\s+\w+\s*:\s*pass/,
34
+ languages: ['python'],
35
+ },
36
+ // 가짜 성공
37
+ {
38
+ severity: 'critical',
39
+ category: 'fake_execution',
40
+ message: '하드코딩된 성공 반환 — 실제 로직 없이 true/ok 반환',
41
+ pattern: /return\s+(?:true|'ok'|"ok"|'success'|"success")\s*;?\s*\/\/\s*(?:always|todo|fixme|hack)/i,
42
+ },
43
+ // 하드코딩 시크릿
44
+ {
45
+ severity: 'critical',
46
+ category: 'hardcoded_secret',
47
+ message: '하드코딩된 비밀키/토큰 패턴',
48
+ pattern: /(?:password|secret|api_key|apikey|token|private_key)\s*[:=]\s*['"][^'"]{8,}['"]/i,
49
+ excludeIf: (line, fp) => isTestPath(fp) || /example|sample|template|placeholder|dummy|config\.example/i.test(fp) || /process\.env|env\.|getenv|os\.environ/i.test(line) || /token:\s*'[a-z_]+'/.test(line),
50
+ },
51
+ // 디버그 코드 잔류
52
+ {
53
+ severity: 'critical',
54
+ category: 'debug_leftover',
55
+ message: 'debugger 문 잔류',
56
+ pattern: /^\s*debugger\s*;?\s*$/,
57
+ languages: ['typescript', 'javascript'],
58
+ },
59
+ // ============ WARNING ============
60
+ // TODO + 빈 구현
61
+ {
62
+ severity: 'warning',
63
+ category: 'incomplete',
64
+ message: 'TODO/FIXME + 빈 구현 (pass/return/throw)',
65
+ pattern: /(?:\/\/|#)\s*(?:TODO|FIXME|XXX|HACK)\b.*$/i,
66
+ },
67
+ // console.log 남발 (테스트 제외)
68
+ {
69
+ severity: 'warning',
70
+ category: 'debug_leftover',
71
+ message: 'console.log 잔류 — 프로덕션 코드에 디버그 출력',
72
+ pattern: /console\.log\s*\(/,
73
+ languages: ['typescript', 'javascript'],
74
+ excludeIf: (line, fp) => isTestPath(fp) ||
75
+ /scripts\//.test(fp) ||
76
+ /cli\/|cli\.ts|runners\//.test(fp) || // CLI 도구 — console.log는 사용자 출력
77
+ /support\/chat|support\/chatTui/.test(fp) || // TUI — console.log는 UI 출력
78
+ /index\.ts$/.test(fp) || // 서비스 진입점
79
+ /core\/service/.test(fp) || // 서비스 로깅
80
+ /discord\//.test(fp) || // Discord 핸들러 로깅
81
+ /memory\//.test(fp) || // 메모리 시스템 로깅
82
+ /automation\//.test(fp) || // 자동화 시스템 로깅
83
+ /console\.(?:warn|error|info)/.test(line) ||
84
+ /\[\w+\]/.test(line) || /`\[/.test(line) || /\$\{.*(?:taskPrefix|prefix)/.test(line),
85
+ },
86
+ // as any (TypeScript)
87
+ {
88
+ severity: 'warning',
89
+ category: 'type_safety',
90
+ message: 'as any — 타입 안전성 우회',
91
+ pattern: /as\s+any\b/,
92
+ languages: ['typescript'],
93
+ excludeIf: (line, fp) => isTestPath(fp) || /eslint-disable/.test(line) ||
94
+ /JSON\.parse|URLSearchParams|\.map\(/.test(line), // JSON/API 결과 캐스트
95
+ },
96
+ // any 타입 사용
97
+ {
98
+ severity: 'warning',
99
+ category: 'type_safety',
100
+ message: ': any 타입 사용 — 타입 안전성 부재',
101
+ pattern: /:\s*any\b(?!\[)/,
102
+ languages: ['typescript'],
103
+ excludeIf: (line, fp) => isTestPath(fp) || /eslint-disable/.test(line) || /\/\//.test(line.split(':')[0] ?? '') ||
104
+ /function\s+normalize|parsed:\s*any|JSON\.parse|response\.\w+\.map/.test(line), // JSON 파싱 결과 접근 패턴
105
+ },
106
+ // non-null assertion
107
+ {
108
+ severity: 'warning',
109
+ category: 'type_safety',
110
+ message: 'non-null assertion (!) — null 체크 없이 단언',
111
+ pattern: /\w+!\./,
112
+ languages: ['typescript'],
113
+ excludeIf: (_line, fp) => isTestPath(fp),
114
+ },
115
+ // eval 사용
116
+ {
117
+ severity: 'warning',
118
+ category: 'security',
119
+ message: 'eval() 사용 — 코드 인젝션 위험',
120
+ pattern: /\beval\s*\(/,
121
+ excludeIf: (_line, fp) => isTestPath(fp),
122
+ },
123
+ // fake API / example.com
124
+ {
125
+ severity: 'warning',
126
+ category: 'fake_data',
127
+ message: 'example.com 등 가짜 URL — 실 운영 불가',
128
+ pattern: /(?:example\.com|localhost:\d{4}|127\.0\.0\.1:\d{4})/,
129
+ excludeIf: (line, fp) => isTestPath(fp) || /\/\//.test(line.split('http')[0] ?? '') || /config|env|default|Html\.ts|linear|github/i.test(fp) || /config|env|default/i.test(line),
130
+ },
131
+ // ============ MINOR ============
132
+ // 매직 넘버
133
+ {
134
+ severity: 'minor',
135
+ category: 'magic_number',
136
+ message: '매직 넘버 — 의미 불명확한 하드코딩 숫자',
137
+ pattern: /(?:=|return|>|<|===|!==)\s*(?:(?!0\b|1\b|2\b|-1\b|100\b|1000\b|60\b|24\b|365\b)\d{3,})\b/,
138
+ excludeIf: (line, fp) => isTestPath(fp) || /const|timeout|limit|max|min|port|size|width|height|delay|interval|ms|sec/i.test(line),
139
+ },
140
+ // 긴 줄 (200자 이상)
141
+ {
142
+ severity: 'minor',
143
+ category: 'readability',
144
+ message: '200자 이상 긴 줄 — 가독성 저하',
145
+ pattern: /.{200,}/,
146
+ excludeIf: (line, _fp) => /^[\s]*\/\//.test(line) || /^[\s]*\*/.test(line) || /import\s/.test(line) || /https?:\/\//.test(line),
147
+ },
148
+ ];
149
+ // ============ 언어 감지 ============
150
+ function detectLanguageForBs(ext) {
151
+ const map = {
152
+ '.ts': 'typescript', '.tsx': 'typescript', '.js': 'javascript', '.jsx': 'javascript',
153
+ '.mjs': 'javascript', '.cjs': 'javascript',
154
+ '.py': 'python', '.pyw': 'python',
155
+ '.go': 'go', '.rs': 'rust', '.java': 'java',
156
+ '.c': 'c', '.h': 'c', '.cpp': 'cpp', '.cxx': 'cpp', '.cc': 'cpp',
157
+ '.hpp': 'cpp', '.hxx': 'cpp', '.cs': 'csharp',
158
+ };
159
+ return map[ext] ?? null;
160
+ }
161
+ // ============ 단일 파일 스캔 ============
162
+ export function scanFileContent(content, filePath, language) {
163
+ const issues = [];
164
+ const lines = content.split('\n');
165
+ for (let i = 0; i < lines.length; i++) {
166
+ const line = lines[i];
167
+ for (const bp of BS_PATTERNS) {
168
+ // 언어 필터
169
+ if (bp.languages && !bp.languages.includes(language))
170
+ continue;
171
+ const match = line.match(bp.pattern);
172
+ if (!match)
173
+ continue;
174
+ // 예외 필터
175
+ if (bp.excludeIf && bp.excludeIf(line, filePath))
176
+ continue;
177
+ issues.push({
178
+ severity: bp.severity,
179
+ category: bp.category,
180
+ message: bp.message,
181
+ filePath,
182
+ line: i + 1,
183
+ matchedText: match[0].slice(0, 80),
184
+ });
185
+ }
186
+ }
187
+ return issues;
188
+ }
189
+ // ============ 파일 스캔 (비동기) ============
190
+ export async function scanFile(filePath) {
191
+ const ext = extname(filePath);
192
+ const language = detectLanguageForBs(ext);
193
+ if (!language)
194
+ return [];
195
+ const content = await readFile(filePath, 'utf-8');
196
+ return scanFileContent(content, filePath, language);
197
+ }
198
+ // ============ 결과 집계 ============
199
+ export function aggregateResults(issues, filesScanned) {
200
+ const critical = issues.filter(i => i.severity === 'critical').length;
201
+ const warning = issues.filter(i => i.severity === 'warning').length;
202
+ const minor = issues.filter(i => i.severity === 'minor').length;
203
+ const bsScore = filesScanned > 0
204
+ ? (critical * 10 + warning * 3 + minor * 1) / filesScanned
205
+ : 0;
206
+ return { issues, filesScanned, critical, warning, minor, bsScore };
207
+ }
208
+ // ============ 레포 전체 스캔 ============
209
+ import { readdir, stat } from 'node:fs/promises';
210
+ import { join } from 'node:path';
211
+ const SKIP_DIRS = new Set([
212
+ 'node_modules', '.git', 'dist', 'build', '__pycache__',
213
+ '.next', '.venv', 'venv', 'coverage', '.turbo', '.cache',
214
+ 'trash', 'testing', 'vendor', 'third_party', 'target',
215
+ 'bin', 'obj', '.openswarm',
216
+ ]);
217
+ const SOURCE_EXTENSIONS = new Set([
218
+ '.ts', '.tsx', '.js', '.jsx', '.mjs', '.cjs',
219
+ '.py', '.pyw', '.go', '.rs', '.java',
220
+ '.c', '.h', '.cpp', '.cxx', '.cc', '.hpp', '.cs',
221
+ ]);
222
+ const MAX_FILE_SIZE = 512 * 1024;
223
+ export async function scanRepository(projectPath, options) {
224
+ const verbose = options?.verbose ?? false;
225
+ const allIssues = [];
226
+ let filesScanned = 0;
227
+ async function walk(dirPath, relPath) {
228
+ let entries;
229
+ try {
230
+ entries = await readdir(dirPath, { withFileTypes: true });
231
+ }
232
+ catch {
233
+ return;
234
+ }
235
+ for (const entry of entries) {
236
+ const fullPath = join(dirPath, entry.name);
237
+ const entryRelPath = relPath ? `${relPath}/${entry.name}` : entry.name;
238
+ if (entry.isDirectory()) {
239
+ if (SKIP_DIRS.has(entry.name))
240
+ continue;
241
+ await walk(fullPath, entryRelPath);
242
+ }
243
+ else if (entry.isFile()) {
244
+ const ext = extname(entry.name);
245
+ if (!SOURCE_EXTENSIONS.has(ext))
246
+ continue;
247
+ const language = detectLanguageForBs(ext);
248
+ if (!language)
249
+ continue;
250
+ try {
251
+ const fileStat = await stat(fullPath);
252
+ if (fileStat.size > MAX_FILE_SIZE)
253
+ continue;
254
+ }
255
+ catch {
256
+ continue;
257
+ }
258
+ try {
259
+ const content = await readFile(fullPath, 'utf-8');
260
+ const issues = scanFileContent(content, entryRelPath, language);
261
+ allIssues.push(...issues);
262
+ filesScanned++;
263
+ if (verbose && issues.length > 0) {
264
+ console.log(` [bs] ${entryRelPath}: ${issues.length} issues`);
265
+ }
266
+ }
267
+ catch {
268
+ continue;
269
+ }
270
+ }
271
+ }
272
+ }
273
+ await walk(projectPath, '');
274
+ return aggregateResults(allIssues, filesScanned);
275
+ }
276
+ //# sourceMappingURL=bsDetector.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bsDetector.js","sourceRoot":"","sources":["../../src/registry/bsDetector.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,36 @@
1
+ import type { EntityKind } from './schema.js';
2
+ type Language = 'typescript' | 'python' | 'go' | 'rust' | 'java' | 'c' | 'cpp' | 'csharp';
3
+ export interface ExtractedEntity {
4
+ kind: EntityKind;
5
+ name: string;
6
+ filePath: string;
7
+ lineStart: number;
8
+ lineEnd?: number;
9
+ signature?: string;
10
+ isExported: boolean;
11
+ loc: number;
12
+ nestingDepth: number;
13
+ paramCount: number;
14
+ }
15
+ /**
16
+ * 소스 파일 내용에서 엔티티 선언을 추출
17
+ */
18
+ export declare function extractEntities(content: string, filePath: string, language: Language): ExtractedEntity[];
19
+ export interface ScanResult {
20
+ scanned: number;
21
+ extracted: number;
22
+ registered: number;
23
+ updated: number;
24
+ removed: number;
25
+ testsMapped: number;
26
+ errors: string[];
27
+ durationMs: number;
28
+ languageBreakdown: Record<string, number>;
29
+ }
30
+ export declare function scanRepository(projectPath: string, projectId: string, options?: {
31
+ maxDepth?: number;
32
+ timeoutMs?: number;
33
+ verbose?: boolean;
34
+ }): Promise<ScanResult>;
35
+ export {};
36
+ //# sourceMappingURL=entityScanner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"entityScanner.d.ts","sourceRoot":"","sources":["../../src/registry/entityScanner.ts"],"names":[],"mappings":"AAWA,OAAO,KAAK,EAAE,UAAU,EAAa,MAAM,aAAa,CAAC;AAsBzD,KAAK,QAAQ,GAAG,YAAY,GAAG,QAAQ,GAAG,IAAI,GAAG,MAAM,GAAG,MAAM,GAAG,GAAG,GAAG,KAAK,GAAG,QAAQ,CAAC;AA4O1F,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,UAAU,CAAC;IACjB,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,OAAO,CAAC;IACpB,GAAG,EAAE,MAAM,CAAC;IACZ,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,MAAM,CAAC;CACpB;AAID;;GAEG;AACH,wBAAgB,eAAe,CAC7B,OAAO,EAAE,MAAM,EACf,QAAQ,EAAE,MAAM,EAChB,QAAQ,EAAE,QAAQ,GACjB,eAAe,EAAE,CAmEnB;AAwPD,MAAM,WAAW,UAAU;IACzB,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,MAAM,EAAE,CAAC;IACjB,UAAU,EAAE,MAAM,CAAC;IACnB,iBAAiB,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CAC3C;AAID,wBAAsB,cAAc,CAClC,WAAW,EAAE,MAAM,EACnB,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE;IAAE,QAAQ,CAAC,EAAE,MAAM,CAAC;IAAC,SAAS,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,OAAO,CAAA;CAAE,GACrE,OAAO,CAAC,UAAU,CAAC,CA0KrB"}