projscan 0.1.13 → 0.2.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 (38) hide show
  1. package/README.md +92 -9
  2. package/dist/cli/index.js +108 -144
  3. package/dist/cli/index.js.map +1 -1
  4. package/dist/core/fileInspector.d.ts +13 -0
  5. package/dist/core/fileInspector.js +205 -0
  6. package/dist/core/fileInspector.js.map +1 -0
  7. package/dist/core/hotspotAnalyzer.d.ts +16 -0
  8. package/dist/core/hotspotAnalyzer.js +342 -0
  9. package/dist/core/hotspotAnalyzer.js.map +1 -0
  10. package/dist/index.d.ts +7 -1
  11. package/dist/index.js +6 -0
  12. package/dist/index.js.map +1 -1
  13. package/dist/mcp/prompts.d.ts +14 -0
  14. package/dist/mcp/prompts.js +126 -0
  15. package/dist/mcp/prompts.js.map +1 -0
  16. package/dist/mcp/resources.d.ts +8 -0
  17. package/dist/mcp/resources.js +57 -0
  18. package/dist/mcp/resources.js.map +1 -0
  19. package/dist/mcp/server.d.ts +5 -0
  20. package/dist/mcp/server.js +205 -0
  21. package/dist/mcp/server.js.map +1 -0
  22. package/dist/mcp/tools.d.ts +9 -0
  23. package/dist/mcp/tools.js +176 -0
  24. package/dist/mcp/tools.js.map +1 -0
  25. package/dist/reporters/consoleReporter.d.ts +3 -1
  26. package/dist/reporters/consoleReporter.js +127 -0
  27. package/dist/reporters/consoleReporter.js.map +1 -1
  28. package/dist/reporters/jsonReporter.d.ts +3 -1
  29. package/dist/reporters/jsonReporter.js +6 -0
  30. package/dist/reporters/jsonReporter.js.map +1 -1
  31. package/dist/reporters/markdownReporter.d.ts +3 -1
  32. package/dist/reporters/markdownReporter.js +99 -0
  33. package/dist/reporters/markdownReporter.js.map +1 -1
  34. package/dist/types.d.ts +88 -0
  35. package/dist/utils/baseline.d.ts +4 -4
  36. package/dist/utils/baseline.js +71 -5
  37. package/dist/utils/baseline.js.map +1 -1
  38. package/package.json +2 -2
@@ -0,0 +1,205 @@
1
+ import fs from 'node:fs/promises';
2
+ import path from 'node:path';
3
+ import { scanRepository } from './repositoryScanner.js';
4
+ import { collectIssues } from './issueEngine.js';
5
+ import { analyzeHotspots } from './hotspotAnalyzer.js';
6
+ export async function inspectFile(rootPath, relOrAbsFile, options = {}) {
7
+ const resolvedRoot = path.resolve(rootPath);
8
+ const absolutePath = path.isAbsolute(relOrAbsFile)
9
+ ? relOrAbsFile
10
+ : path.resolve(resolvedRoot, relOrAbsFile);
11
+ if (!isInsideRoot(absolutePath, resolvedRoot)) {
12
+ return makeEmpty(relOrAbsFile, 'File is outside the project root');
13
+ }
14
+ let content;
15
+ let sizeBytes;
16
+ try {
17
+ const stat = await fs.stat(absolutePath);
18
+ if (!stat.isFile()) {
19
+ return makeEmpty(relOrAbsFile, 'Path is not a file');
20
+ }
21
+ sizeBytes = stat.size;
22
+ content = await fs.readFile(absolutePath, 'utf-8');
23
+ }
24
+ catch (err) {
25
+ const msg = err.code === 'ENOENT' ? 'File not found' : String(err);
26
+ return makeEmpty(relOrAbsFile, msg);
27
+ }
28
+ const relativePath = path.relative(resolvedRoot, absolutePath).split(path.sep).join('/');
29
+ const lines = content.split('\n');
30
+ const imports = extractImports(content);
31
+ const exports = extractExports(content);
32
+ const purpose = inferPurpose(absolutePath, exports);
33
+ const potentialIssues = detectFileIssues(content, lines.length);
34
+ const files = options.scan?.files ?? (await scanRepository(resolvedRoot)).files;
35
+ const issues = options.issues ?? (await collectIssues(resolvedRoot, files));
36
+ const hotspotReport = options.hotspots ?? (await analyzeHotspots(resolvedRoot, files, issues, { limit: 100 }));
37
+ const hotspot = findHotspotForFile(hotspotReport, relativePath);
38
+ const relatedIssues = issues.filter((issue) => (issue.title + '\n' + issue.description).includes(relativePath));
39
+ return {
40
+ relativePath,
41
+ exists: true,
42
+ purpose,
43
+ lineCount: lines.length,
44
+ sizeBytes,
45
+ imports,
46
+ exports,
47
+ potentialIssues,
48
+ hotspot,
49
+ issues: relatedIssues,
50
+ };
51
+ }
52
+ function makeEmpty(relativePath, reason) {
53
+ return {
54
+ relativePath,
55
+ exists: false,
56
+ reason,
57
+ purpose: '',
58
+ lineCount: 0,
59
+ sizeBytes: 0,
60
+ imports: [],
61
+ exports: [],
62
+ potentialIssues: [],
63
+ hotspot: null,
64
+ issues: [],
65
+ };
66
+ }
67
+ function isInsideRoot(absolutePath, resolvedRoot) {
68
+ return absolutePath === resolvedRoot || absolutePath.startsWith(resolvedRoot + path.sep);
69
+ }
70
+ function findHotspotForFile(report, relativePath) {
71
+ if (!report || !report.available)
72
+ return null;
73
+ return report.hotspots.find((h) => h.relativePath === relativePath) ?? null;
74
+ }
75
+ // ── Parsing (shared with CLI/MCP) ─────────────────────────
76
+ export function extractImports(content) {
77
+ const imports = [];
78
+ const seen = new Set();
79
+ const esImportRegex = /import\s+(?:(?:\{[^}]*\}|[\w*]+(?:\s*,\s*\{[^}]*\})?|\*\s+as\s+\w+)\s+from\s+)?['"]([^'"]+)['"]/gm;
80
+ let match;
81
+ while ((match = esImportRegex.exec(content)) !== null) {
82
+ const source = match[1];
83
+ if (!seen.has(source)) {
84
+ seen.add(source);
85
+ imports.push({
86
+ source,
87
+ specifiers: [],
88
+ isRelative: source.startsWith('.') || source.startsWith('/'),
89
+ });
90
+ }
91
+ }
92
+ const requireRegex = /(?:const|let|var)\s+(?:\{[^}]*\}|\w+)\s*=\s*require\(\s*['"]([^'"]+)['"]\s*\)/gm;
93
+ while ((match = requireRegex.exec(content)) !== null) {
94
+ const source = match[1];
95
+ if (!seen.has(source)) {
96
+ seen.add(source);
97
+ imports.push({
98
+ source,
99
+ specifiers: [],
100
+ isRelative: source.startsWith('.') || source.startsWith('/'),
101
+ });
102
+ }
103
+ }
104
+ return imports;
105
+ }
106
+ export function extractExports(content) {
107
+ const exports = [];
108
+ const funcRegex = /^export\s+(?:async\s+)?function\s+(\w+)/gm;
109
+ let match;
110
+ while ((match = funcRegex.exec(content)) !== null) {
111
+ exports.push({ name: match[1], type: 'function' });
112
+ }
113
+ const classRegex = /^export\s+class\s+(\w+)/gm;
114
+ while ((match = classRegex.exec(content)) !== null) {
115
+ exports.push({ name: match[1], type: 'class' });
116
+ }
117
+ const varRegex = /^export\s+(?:const|let|var)\s+(\w+)/gm;
118
+ while ((match = varRegex.exec(content)) !== null) {
119
+ exports.push({ name: match[1], type: 'variable' });
120
+ }
121
+ const interfaceRegex = /^export\s+interface\s+(\w+)/gm;
122
+ while ((match = interfaceRegex.exec(content)) !== null) {
123
+ exports.push({ name: match[1], type: 'interface' });
124
+ }
125
+ const typeRegex = /^export\s+type\s+(\w+)/gm;
126
+ while ((match = typeRegex.exec(content)) !== null) {
127
+ exports.push({ name: match[1], type: 'type' });
128
+ }
129
+ if (/^export\s+default/m.test(content)) {
130
+ exports.push({ name: 'default', type: 'default' });
131
+ }
132
+ return exports;
133
+ }
134
+ export function inferPurpose(filePath, exports) {
135
+ const name = path.basename(filePath, path.extname(filePath)).toLowerCase();
136
+ const dir = path.dirname(filePath).toLowerCase();
137
+ if (name.includes('test') || name.includes('spec'))
138
+ return 'Test file';
139
+ if (name.includes('config') || name.includes('rc'))
140
+ return 'Configuration file';
141
+ if (name === 'index')
142
+ return 'Module entry point / barrel file';
143
+ if (name === 'main' || name === 'app')
144
+ return 'Application entry point';
145
+ if (name.includes('route') || name.includes('router'))
146
+ return 'Route definitions';
147
+ if (name.includes('middleware'))
148
+ return 'Middleware handler';
149
+ if (name.includes('controller'))
150
+ return 'Request controller';
151
+ if (name.includes('service'))
152
+ return 'Service layer logic';
153
+ if (name.includes('model') || name.includes('schema'))
154
+ return 'Data model / schema definition';
155
+ if (name.includes('util') || name.includes('helper'))
156
+ return 'Utility functions';
157
+ if (name.includes('hook'))
158
+ return 'Custom hook';
159
+ if (name.includes('context') || name.includes('provider'))
160
+ return 'Context / state provider';
161
+ if (name.includes('type') || name.includes('interface'))
162
+ return 'Type definitions';
163
+ if (name.includes('constant'))
164
+ return 'Constants / configuration';
165
+ if (name.includes('migration'))
166
+ return 'Database migration';
167
+ if (name.includes('seed'))
168
+ return 'Database seed data';
169
+ if (name.includes('auth'))
170
+ return 'Authentication logic';
171
+ if (name.includes('api'))
172
+ return 'API endpoint handler';
173
+ if (dir.includes('component') || dir.includes('pages'))
174
+ return 'UI component';
175
+ if (dir.includes('service'))
176
+ return 'Service module';
177
+ if (dir.includes('model'))
178
+ return 'Data model';
179
+ if (dir.includes('util') || dir.includes('lib'))
180
+ return 'Library / utility module';
181
+ const exportTypes = exports.map((e) => e.type);
182
+ if (exportTypes.includes('class'))
183
+ return 'Class-based module';
184
+ if (exportTypes.filter((t) => t === 'function').length > 2)
185
+ return 'Function library';
186
+ return 'Source module';
187
+ }
188
+ export function detectFileIssues(content, lineCount) {
189
+ const issues = [];
190
+ if (lineCount > 500)
191
+ issues.push(`Large file (${lineCount} lines) — consider splitting`);
192
+ if (lineCount > 1000)
193
+ issues.push('Very large file — strongly consider refactoring');
194
+ if (/console\.(log|warn|error|debug)\s*\(/.test(content)) {
195
+ issues.push('Contains console.log statements — consider using a proper logger');
196
+ }
197
+ if (/TODO|FIXME|HACK|XXX/i.test(content)) {
198
+ issues.push('Contains TODO/FIXME comments');
199
+ }
200
+ if (/:\s*any\b/.test(content) && /\.tsx?$/.test(content)) {
201
+ issues.push('Uses "any" type — consider using proper types');
202
+ }
203
+ return issues;
204
+ }
205
+ //# sourceMappingURL=fileInspector.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fileInspector.js","sourceRoot":"","sources":["../../src/core/fileInspector.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,16 @@
1
+ import type { FileEntry, Issue, HotspotReport } from '../types.js';
2
+ export interface HotspotOptions {
3
+ since?: string;
4
+ limit?: number;
5
+ }
6
+ export declare function analyzeHotspots(rootPath: string, files: FileEntry[], issues: Issue[], options?: HotspotOptions): Promise<HotspotReport>;
7
+ interface ScoreInputs {
8
+ churn: number;
9
+ lines: number;
10
+ authors: number;
11
+ daysSinceLastChange: number | null;
12
+ issueCount: number;
13
+ busFactorOne?: boolean;
14
+ }
15
+ export declare function computeRiskScore(i: ScoreInputs): number;
16
+ export {};
@@ -0,0 +1,342 @@
1
+ import { spawn } from 'node:child_process';
2
+ import fs from 'node:fs/promises';
3
+ import path from 'node:path';
4
+ const CODE_EXTENSIONS = new Set([
5
+ '.ts', '.tsx', '.js', '.jsx', '.mjs', '.cjs',
6
+ '.py', '.rb', '.go', '.java', '.rs', '.php',
7
+ '.c', '.cc', '.cpp', '.h', '.hpp',
8
+ '.cs', '.swift', '.kt', '.kts', '.scala',
9
+ '.vue', '.svelte',
10
+ ]);
11
+ const MAX_LINE_READS = 400;
12
+ const MAX_LINE_READ_BYTES = 512 * 1024;
13
+ const DEFAULT_SINCE = '12 months ago';
14
+ export async function analyzeHotspots(rootPath, files, issues, options = {}) {
15
+ const limit = Math.max(1, Math.min(100, options.limit ?? 10));
16
+ const since = options.since ?? DEFAULT_SINCE;
17
+ const isRepo = await isGitRepository(rootPath);
18
+ if (!isRepo) {
19
+ return {
20
+ available: false,
21
+ reason: 'Not a git repository — hotspot analysis requires git history',
22
+ window: { since: null, commitsScanned: 0 },
23
+ hotspots: [],
24
+ totalFilesRanked: 0,
25
+ };
26
+ }
27
+ const churnMap = await collectGitChurn(rootPath, since);
28
+ const candidates = files.filter((f) => CODE_EXTENSIONS.has(f.extension) && f.sizeBytes <= MAX_LINE_READ_BYTES);
29
+ const candidatesByScore = [...candidates].sort((a, b) => {
30
+ const ca = churnMap.get(a.relativePath)?.churn ?? 0;
31
+ const cb = churnMap.get(b.relativePath)?.churn ?? 0;
32
+ if (cb !== ca)
33
+ return cb - ca;
34
+ return b.sizeBytes - a.sizeBytes;
35
+ });
36
+ const readTargets = candidatesByScore.slice(0, MAX_LINE_READS);
37
+ const lineCounts = new Map();
38
+ await Promise.all(readTargets.map(async (f) => {
39
+ const lines = await countLines(f.absolutePath);
40
+ if (lines !== null)
41
+ lineCounts.set(f.relativePath, lines);
42
+ }));
43
+ const issueIndex = indexIssuesByFile(issues, files);
44
+ const now = Date.now();
45
+ const hotspots = candidates.map((file) => {
46
+ const churnEntry = churnMap.get(file.relativePath);
47
+ const churn = churnEntry?.churn ?? 0;
48
+ const authors = churnEntry?.authors.size ?? 0;
49
+ const lastTs = churnEntry?.lastTimestampMs ?? null;
50
+ const daysSinceLastChange = lastTs === null ? null : Math.max(0, Math.floor((now - lastTs) / (1000 * 60 * 60 * 24)));
51
+ const lines = lineCounts.get(file.relativePath) ?? estimateLines(file.sizeBytes);
52
+ const issueIds = issueIndex.get(file.relativePath) ?? [];
53
+ const topAuthors = rankAuthors(churnEntry?.authorCommits);
54
+ const primaryAuthor = topAuthors[0]?.author ?? null;
55
+ const primaryAuthorShare = topAuthors[0]?.share ?? 0;
56
+ const busFactorOne = churn >= 3 && primaryAuthorShare >= 0.8;
57
+ const riskScore = computeRiskScore({
58
+ churn,
59
+ lines,
60
+ authors,
61
+ daysSinceLastChange,
62
+ issueCount: issueIds.length,
63
+ busFactorOne,
64
+ });
65
+ const reasons = buildReasons({
66
+ churn,
67
+ lines,
68
+ authors,
69
+ daysSinceLastChange,
70
+ issueCount: issueIds.length,
71
+ busFactorOne,
72
+ primaryAuthor,
73
+ });
74
+ return {
75
+ relativePath: file.relativePath,
76
+ churn,
77
+ distinctAuthors: authors,
78
+ daysSinceLastChange,
79
+ lineCount: lines,
80
+ sizeBytes: file.sizeBytes,
81
+ issueCount: issueIds.length,
82
+ issueIds,
83
+ riskScore,
84
+ reasons,
85
+ primaryAuthor,
86
+ primaryAuthorShare,
87
+ busFactorOne,
88
+ topAuthors,
89
+ };
90
+ });
91
+ hotspots.sort((a, b) => b.riskScore - a.riskScore);
92
+ const ranked = hotspots.filter((h) => h.riskScore > 0);
93
+ return {
94
+ available: true,
95
+ window: {
96
+ since,
97
+ commitsScanned: countCommits(churnMap),
98
+ },
99
+ hotspots: ranked.slice(0, limit),
100
+ totalFilesRanked: ranked.length,
101
+ };
102
+ }
103
+ async function isGitRepository(rootPath) {
104
+ try {
105
+ const { code } = await runGit(rootPath, ['rev-parse', '--is-inside-work-tree']);
106
+ return code === 0;
107
+ }
108
+ catch {
109
+ return false;
110
+ }
111
+ }
112
+ async function collectGitChurn(rootPath, since) {
113
+ const map = new Map();
114
+ const args = [
115
+ 'log',
116
+ `--since=${since}`,
117
+ '--no-merges',
118
+ '--name-only',
119
+ '--pretty=format:COMMIT|%H|%ae|%at',
120
+ ];
121
+ let stdout;
122
+ try {
123
+ const result = await runGit(rootPath, args, { timeoutMs: 15_000 });
124
+ if (result.code !== 0)
125
+ return map;
126
+ stdout = result.stdout;
127
+ }
128
+ catch {
129
+ return map;
130
+ }
131
+ let currentHash = null;
132
+ let currentAuthor = null;
133
+ let currentTsMs = null;
134
+ for (const rawLine of stdout.split('\n')) {
135
+ const line = rawLine.trimEnd();
136
+ if (!line)
137
+ continue;
138
+ if (line.startsWith('COMMIT|')) {
139
+ const parts = line.split('|');
140
+ currentHash = parts[1] ?? null;
141
+ currentAuthor = parts[2] ?? null;
142
+ const tsSec = parseInt(parts[3] ?? '', 10);
143
+ currentTsMs = Number.isFinite(tsSec) ? tsSec * 1000 : null;
144
+ continue;
145
+ }
146
+ if (!currentHash)
147
+ continue;
148
+ const filePath = normalizeFilePath(line);
149
+ if (!filePath)
150
+ continue;
151
+ let entry = map.get(filePath);
152
+ if (!entry) {
153
+ entry = {
154
+ churn: 0,
155
+ authors: new Set(),
156
+ authorCommits: new Map(),
157
+ lastTimestampMs: null,
158
+ commitHashes: new Set(),
159
+ };
160
+ map.set(filePath, entry);
161
+ }
162
+ if (!entry.commitHashes.has(currentHash)) {
163
+ entry.commitHashes.add(currentHash);
164
+ entry.churn++;
165
+ if (currentAuthor) {
166
+ entry.authorCommits.set(currentAuthor, (entry.authorCommits.get(currentAuthor) ?? 0) + 1);
167
+ }
168
+ }
169
+ if (currentAuthor)
170
+ entry.authors.add(currentAuthor);
171
+ if (currentTsMs !== null && (entry.lastTimestampMs === null || currentTsMs > entry.lastTimestampMs)) {
172
+ entry.lastTimestampMs = currentTsMs;
173
+ }
174
+ }
175
+ return map;
176
+ }
177
+ function normalizeFilePath(line) {
178
+ const trimmed = line.trim();
179
+ if (!trimmed)
180
+ return null;
181
+ return trimmed.split(path.sep).join('/').replace(/^\.\//, '');
182
+ }
183
+ function countCommits(churnMap) {
184
+ const hashes = new Set();
185
+ for (const entry of churnMap.values()) {
186
+ for (const h of entry.commitHashes)
187
+ hashes.add(h);
188
+ }
189
+ return hashes.size;
190
+ }
191
+ function runGit(cwd, args, opts = {}) {
192
+ return new Promise((resolve, reject) => {
193
+ const child = spawn('git', args, { cwd, env: process.env });
194
+ let stdout = '';
195
+ let stderr = '';
196
+ let settled = false;
197
+ const timeout = opts.timeoutMs
198
+ ? setTimeout(() => {
199
+ if (settled)
200
+ return;
201
+ settled = true;
202
+ child.kill('SIGKILL');
203
+ reject(new Error('git command timed out'));
204
+ }, opts.timeoutMs)
205
+ : null;
206
+ child.stdout.on('data', (d) => {
207
+ stdout += d.toString();
208
+ });
209
+ child.stderr.on('data', (d) => {
210
+ stderr += d.toString();
211
+ });
212
+ child.on('error', (err) => {
213
+ if (settled)
214
+ return;
215
+ settled = true;
216
+ if (timeout)
217
+ clearTimeout(timeout);
218
+ reject(err);
219
+ });
220
+ child.on('close', (code) => {
221
+ if (settled)
222
+ return;
223
+ settled = true;
224
+ if (timeout)
225
+ clearTimeout(timeout);
226
+ resolve({ code: code ?? 1, stdout, stderr });
227
+ });
228
+ });
229
+ }
230
+ // ── Line Counting ─────────────────────────────────────────
231
+ async function countLines(absolutePath) {
232
+ try {
233
+ const content = await fs.readFile(absolutePath, 'utf-8');
234
+ if (!content)
235
+ return 0;
236
+ let lines = 1;
237
+ for (let i = 0; i < content.length; i++) {
238
+ if (content.charCodeAt(i) === 10)
239
+ lines++;
240
+ }
241
+ return lines;
242
+ }
243
+ catch {
244
+ return null;
245
+ }
246
+ }
247
+ function estimateLines(sizeBytes) {
248
+ return Math.max(1, Math.round(sizeBytes / 40));
249
+ }
250
+ // ── Issue → File Mapping ──────────────────────────────────
251
+ function indexIssuesByFile(issues, files) {
252
+ const index = new Map();
253
+ const filePaths = new Set(files.map((f) => f.relativePath));
254
+ for (const issue of issues) {
255
+ const haystack = `${issue.title}\n${issue.description}`;
256
+ for (const filePath of filePaths) {
257
+ if (!filePath)
258
+ continue;
259
+ if (haystack.includes(filePath)) {
260
+ const list = index.get(filePath) ?? [];
261
+ if (!list.includes(issue.id))
262
+ list.push(issue.id);
263
+ index.set(filePath, list);
264
+ }
265
+ }
266
+ }
267
+ return index;
268
+ }
269
+ export function computeRiskScore(i) {
270
+ const churnWeight = Math.log2(1 + i.churn) * 20;
271
+ const complexityWeight = Math.log2(1 + i.lines) * 4;
272
+ const hotChurnXComplexity = Math.log2(1 + i.churn) * Math.log2(1 + i.lines) * 3;
273
+ const authorWeight = Math.log2(1 + i.authors) * 5;
274
+ const issueWeight = i.issueCount * 12;
275
+ const busFactorPenalty = i.busFactorOne ? 15 : 0;
276
+ let recencyBoost = 0;
277
+ if (i.daysSinceLastChange !== null) {
278
+ if (i.daysSinceLastChange <= 7)
279
+ recencyBoost = 10;
280
+ else if (i.daysSinceLastChange <= 30)
281
+ recencyBoost = 6;
282
+ else if (i.daysSinceLastChange <= 90)
283
+ recencyBoost = 3;
284
+ }
285
+ const raw = churnWeight +
286
+ complexityWeight +
287
+ hotChurnXComplexity +
288
+ authorWeight +
289
+ issueWeight +
290
+ recencyBoost +
291
+ busFactorPenalty;
292
+ if (i.churn === 0 && i.issueCount === 0)
293
+ return 0;
294
+ return Math.round(raw * 10) / 10;
295
+ }
296
+ function buildReasons(i) {
297
+ const reasons = [];
298
+ if (i.churn >= 20)
299
+ reasons.push(`high churn (${i.churn} commits)`);
300
+ else if (i.churn >= 8)
301
+ reasons.push(`frequent changes (${i.churn} commits)`);
302
+ else if (i.churn > 0)
303
+ reasons.push(`${i.churn} commit${i.churn === 1 ? '' : 's'}`);
304
+ if (i.lines >= 500)
305
+ reasons.push(`large file (${i.lines} lines)`);
306
+ else if (i.lines >= 250)
307
+ reasons.push(`${i.lines} lines`);
308
+ if (i.authors >= 5)
309
+ reasons.push(`${i.authors} contributors`);
310
+ else if (i.authors >= 2)
311
+ reasons.push(`${i.authors} contributors`);
312
+ if (i.issueCount > 0) {
313
+ reasons.push(`${i.issueCount} open issue${i.issueCount === 1 ? '' : 's'}`);
314
+ }
315
+ if (i.daysSinceLastChange !== null && i.daysSinceLastChange <= 7) {
316
+ reasons.push('changed this week');
317
+ }
318
+ if (i.busFactorOne && i.primaryAuthor) {
319
+ reasons.push(`bus factor 1 (${formatAuthor(i.primaryAuthor)})`);
320
+ }
321
+ return reasons;
322
+ }
323
+ function rankAuthors(authorCommits) {
324
+ if (!authorCommits || authorCommits.size === 0)
325
+ return [];
326
+ const total = [...authorCommits.values()].reduce((sum, n) => sum + n, 0);
327
+ if (total === 0)
328
+ return [];
329
+ return [...authorCommits.entries()]
330
+ .sort((a, b) => b[1] - a[1])
331
+ .slice(0, 5)
332
+ .map(([author, commits]) => ({
333
+ author,
334
+ commits,
335
+ share: Math.round((commits / total) * 1000) / 1000,
336
+ }));
337
+ }
338
+ function formatAuthor(email) {
339
+ const atIdx = email.indexOf('@');
340
+ return atIdx > 0 ? email.slice(0, atIdx) : email;
341
+ }
342
+ //# sourceMappingURL=hotspotAnalyzer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hotspotAnalyzer.js","sourceRoot":"","sources":["../../src/core/hotspotAnalyzer.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAClC,OAAO,IAAI,MAAM,WAAW,CAAC;AAG7B,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC;IAC9B,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM;IAC5C,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM;IAC3C,IAAI,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;IACjC,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ;IACxC,MAAM,EAAE,SAAS;CAClB,CAAC,CAAC;AAEH,MAAM,cAAc,GAAG,GAAG,CAAC;AAC3B,MAAM,mBAAmB,GAAG,GAAG,GAAG,IAAI,CAAC;AACvC,MAAM,aAAa,GAAG,eAAe,CAAC;AAOtC,MAAM,CAAC,KAAK,UAAU,eAAe,CACnC,QAAgB,EAChB,KAAkB,EAClB,MAAe,EACf,UAA0B,EAAE;IAE5B,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,CAAC;IAC9D,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,IAAI,aAAa,CAAC;IAE7C,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,QAAQ,CAAC,CAAC;IAC/C,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO;YACL,SAAS,EAAE,KAAK;YAChB,MAAM,EAAE,8DAA8D;YACtE,MAAM,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,cAAc,EAAE,CAAC,EAAE;YAC1C,QAAQ,EAAE,EAAE;YACZ,gBAAgB,EAAE,CAAC;SACpB,CAAC;IACJ,CAAC;IAED,MAAM,QAAQ,GAAG,MAAM,eAAe,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IAExD,MAAM,UAAU,GAAG,KAAK,CAAC,MAAM,CAC7B,CAAC,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,SAAS,IAAI,mBAAmB,CAC9E,CAAC;IAEF,MAAM,iBAAiB,GAAG,CAAC,GAAG,UAAU,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;QACtD,MAAM,EAAE,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,YAAY,CAAC,EAAE,KAAK,IAAI,CAAC,CAAC;QACpD,MAAM,EAAE,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,YAAY,CAAC,EAAE,KAAK,IAAI,CAAC,CAAC;QACpD,IAAI,EAAE,KAAK,EAAE;YAAE,OAAO,EAAE,GAAG,EAAE,CAAC;QAC9B,OAAO,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,SAAS,CAAC;IACnC,CAAC,CAAC,CAAC;IAEH,MAAM,WAAW,GAAG,iBAAiB,CAAC,KAAK,CAAC,CAAC,EAAE,cAAc,CAAC,CAAC;IAC/D,MAAM,UAAU,GAAG,IAAI,GAAG,EAAkB,CAAC;IAC7C,MAAM,OAAO,CAAC,GAAG,CACf,WAAW,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE;QAC1B,MAAM,KAAK,GAAG,MAAM,UAAU,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC;QAC/C,IAAI,KAAK,KAAK,IAAI;YAAE,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;IAC5D,CAAC,CAAC,CACH,CAAC;IAEF,MAAM,UAAU,GAAG,iBAAiB,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;IAEpD,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IACvB,MAAM,QAAQ,GAAkB,UAAU,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;QACtD,MAAM,UAAU,GAAG,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACnD,MAAM,KAAK,GAAG,UAAU,EAAE,KAAK,IAAI,CAAC,CAAC;QACrC,MAAM,OAAO,GAAG,UAAU,EAAE,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;QAC9C,MAAM,MAAM,GAAG,UAAU,EAAE,eAAe,IAAI,IAAI,CAAC;QACnD,MAAM,mBAAmB,GACvB,MAAM,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC;QAE3F,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACjF,MAAM,QAAQ,GAAG,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC;QAEzD,MAAM,UAAU,GAAG,WAAW,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC;QAC1D,MAAM,aAAa,GAAG,UAAU,CAAC,CAAC,CAAC,EAAE,MAAM,IAAI,IAAI,CAAC;QACpD,MAAM,kBAAkB,GAAG,UAAU,CAAC,CAAC,CAAC,EAAE,KAAK,IAAI,CAAC,CAAC;QACrD,MAAM,YAAY,GAAG,KAAK,IAAI,CAAC,IAAI,kBAAkB,IAAI,GAAG,CAAC;QAE7D,MAAM,SAAS,GAAG,gBAAgB,CAAC;YACjC,KAAK;YACL,KAAK;YACL,OAAO;YACP,mBAAmB;YACnB,UAAU,EAAE,QAAQ,CAAC,MAAM;YAC3B,YAAY;SACb,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,YAAY,CAAC;YAC3B,KAAK;YACL,KAAK;YACL,OAAO;YACP,mBAAmB;YACnB,UAAU,EAAE,QAAQ,CAAC,MAAM;YAC3B,YAAY;YACZ,aAAa;SACd,CAAC,CAAC;QAEH,OAAO;YACL,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,KAAK;YACL,eAAe,EAAE,OAAO;YACxB,mBAAmB;YACnB,SAAS,EAAE,KAAK;YAChB,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,UAAU,EAAE,QAAQ,CAAC,MAAM;YAC3B,QAAQ;YACR,SAAS;YACT,OAAO;YACP,aAAa;YACb,kBAAkB;YAClB,YAAY;YACZ,UAAU;SACX,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,SAAS,CAAC,CAAC;IAEnD,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,CAAC;IAEvD,OAAO;QACL,SAAS,EAAE,IAAI;QACf,MAAM,EAAE;YACN,KAAK;YACL,cAAc,EAAE,YAAY,CAAC,QAAQ,CAAC;SACvC;QACD,QAAQ,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC;QAChC,gBAAgB,EAAE,MAAM,CAAC,MAAM;KAChC,CAAC;AACJ,CAAC;AAYD,KAAK,UAAU,eAAe,CAAC,QAAgB;IAC7C,IAAI,CAAC;QACH,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,MAAM,CAAC,QAAQ,EAAE,CAAC,WAAW,EAAE,uBAAuB,CAAC,CAAC,CAAC;QAChF,OAAO,IAAI,KAAK,CAAC,CAAC;IACpB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAED,KAAK,UAAU,eAAe,CAAC,QAAgB,EAAE,KAAa;IAC5D,MAAM,GAAG,GAAG,IAAI,GAAG,EAAsB,CAAC;IAE1C,MAAM,IAAI,GAAG;QACX,KAAK;QACL,WAAW,KAAK,EAAE;QAClB,aAAa;QACb,aAAa;QACb,mCAAmC;KACpC,CAAC;IAEF,IAAI,MAAc,CAAC;IACnB,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,QAAQ,EAAE,IAAI,EAAE,EAAE,SAAS,EAAE,MAAM,EAAE,CAAC,CAAC;QACnE,IAAI,MAAM,CAAC,IAAI,KAAK,CAAC;YAAE,OAAO,GAAG,CAAC;QAClC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC;IACzB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,GAAG,CAAC;IACb,CAAC;IAED,IAAI,WAAW,GAAkB,IAAI,CAAC;IACtC,IAAI,aAAa,GAAkB,IAAI,CAAC;IACxC,IAAI,WAAW,GAAkB,IAAI,CAAC;IAEtC,KAAK,MAAM,OAAO,IAAI,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;QACzC,MAAM,IAAI,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC;QAC/B,IAAI,CAAC,IAAI;YAAE,SAAS;QAEpB,IAAI,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;YAC/B,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC9B,WAAW,GAAG,KAAK,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;YAC/B,aAAa,GAAG,KAAK,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC;YACjC,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,EAAE,EAAE,EAAE,CAAC,CAAC;YAC3C,WAAW,GAAG,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;YAC3D,SAAS;QACX,CAAC;QAED,IAAI,CAAC,WAAW;YAAE,SAAS;QAC3B,MAAM,QAAQ,GAAG,iBAAiB,CAAC,IAAI,CAAC,CAAC;QACzC,IAAI,CAAC,QAAQ;YAAE,SAAS;QAExB,IAAI,KAAK,GAAG,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,KAAK,GAAG;gBACN,KAAK,EAAE,CAAC;gBACR,OAAO,EAAE,IAAI,GAAG,EAAE;gBAClB,aAAa,EAAE,IAAI,GAAG,EAAE;gBACxB,eAAe,EAAE,IAAI;gBACrB,YAAY,EAAE,IAAI,GAAG,EAAE;aACxB,CAAC;YACF,GAAG,CAAC,GAAG,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;QAC3B,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC;YACzC,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;YACpC,KAAK,CAAC,KAAK,EAAE,CAAC;YACd,IAAI,aAAa,EAAE,CAAC;gBAClB,KAAK,CAAC,aAAa,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC,KAAK,CAAC,aAAa,CAAC,GAAG,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;YAC5F,CAAC;QACH,CAAC;QACD,IAAI,aAAa;YAAE,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;QACpD,IAAI,WAAW,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,eAAe,KAAK,IAAI,IAAI,WAAW,GAAG,KAAK,CAAC,eAAe,CAAC,EAAE,CAAC;YACpG,KAAK,CAAC,eAAe,GAAG,WAAW,CAAC;QACtC,CAAC;IACH,CAAC;IAED,OAAO,GAAG,CAAC;AACb,CAAC;AAED,SAAS,iBAAiB,CAAC,IAAY;IACrC,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;IAC5B,IAAI,CAAC,OAAO;QAAE,OAAO,IAAI,CAAC;IAC1B,OAAO,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;AAChE,CAAC;AAED,SAAS,YAAY,CAAC,QAAiC;IACrD,MAAM,MAAM,GAAG,IAAI,GAAG,EAAU,CAAC;IACjC,KAAK,MAAM,KAAK,IAAI,QAAQ,CAAC,MAAM,EAAE,EAAE,CAAC;QACtC,KAAK,MAAM,CAAC,IAAI,KAAK,CAAC,YAAY;YAAE,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IACpD,CAAC;IACD,OAAO,MAAM,CAAC,IAAI,CAAC;AACrB,CAAC;AAQD,SAAS,MAAM,CACb,GAAW,EACX,IAAc,EACd,OAA+B,EAAE;IAEjC,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;QAC5D,IAAI,MAAM,GAAG,EAAE,CAAC;QAChB,IAAI,MAAM,GAAG,EAAE,CAAC;QAChB,IAAI,OAAO,GAAG,KAAK,CAAC;QAEpB,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS;YAC5B,CAAC,CAAC,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,OAAO;oBAAE,OAAO;gBACpB,OAAO,GAAG,IAAI,CAAC;gBACf,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBACtB,MAAM,CAAC,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC,CAAC;YAC7C,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC;YACpB,CAAC,CAAC,IAAI,CAAC;QAET,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE;YAC5B,MAAM,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC;QACzB,CAAC,CAAC,CAAC;QACH,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE;YAC5B,MAAM,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC;QACzB,CAAC,CAAC,CAAC;QACH,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YACxB,IAAI,OAAO;gBAAE,OAAO;YACpB,OAAO,GAAG,IAAI,CAAC;YACf,IAAI,OAAO;gBAAE,YAAY,CAAC,OAAO,CAAC,CAAC;YACnC,MAAM,CAAC,GAAG,CAAC,CAAC;QACd,CAAC,CAAC,CAAC;QACH,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,EAAE;YACzB,IAAI,OAAO;gBAAE,OAAO;YACpB,OAAO,GAAG,IAAI,CAAC;YACf,IAAI,OAAO;gBAAE,YAAY,CAAC,OAAO,CAAC,CAAC;YACnC,OAAO,CAAC,EAAE,IAAI,EAAE,IAAI,IAAI,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AAED,6DAA6D;AAE7D,KAAK,UAAU,UAAU,CAAC,YAAoB;IAC5C,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;QACzD,IAAI,CAAC,OAAO;YAAE,OAAO,CAAC,CAAC;QACvB,IAAI,KAAK,GAAG,CAAC,CAAC;QACd,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YACxC,IAAI,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC,KAAK,EAAE;gBAAE,KAAK,EAAE,CAAC;QAC5C,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,SAAS,aAAa,CAAC,SAAiB;IACtC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC,CAAC;AACjD,CAAC;AAED,6DAA6D;AAE7D,SAAS,iBAAiB,CAAC,MAAe,EAAE,KAAkB;IAC5D,MAAM,KAAK,GAAG,IAAI,GAAG,EAAoB,CAAC;IAC1C,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC;IAE5D,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;QAC3B,MAAM,QAAQ,GAAG,GAAG,KAAK,CAAC,KAAK,KAAK,KAAK,CAAC,WAAW,EAAE,CAAC;QACxD,KAAK,MAAM,QAAQ,IAAI,SAAS,EAAE,CAAC;YACjC,IAAI,CAAC,QAAQ;gBAAE,SAAS;YACxB,IAAI,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;gBAChC,MAAM,IAAI,GAAG,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;gBACvC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;oBAAE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBAClD,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YAC5B,CAAC;QACH,CAAC;IACH,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAaD,MAAM,UAAU,gBAAgB,CAAC,CAAc;IAC7C,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC;IAChD,MAAM,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACpD,MAAM,mBAAmB,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAChF,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAClD,MAAM,WAAW,GAAG,CAAC,CAAC,UAAU,GAAG,EAAE,CAAC;IACtC,MAAM,gBAAgB,GAAG,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;IAEjD,IAAI,YAAY,GAAG,CAAC,CAAC;IACrB,IAAI,CAAC,CAAC,mBAAmB,KAAK,IAAI,EAAE,CAAC;QACnC,IAAI,CAAC,CAAC,mBAAmB,IAAI,CAAC;YAAE,YAAY,GAAG,EAAE,CAAC;aAC7C,IAAI,CAAC,CAAC,mBAAmB,IAAI,EAAE;YAAE,YAAY,GAAG,CAAC,CAAC;aAClD,IAAI,CAAC,CAAC,mBAAmB,IAAI,EAAE;YAAE,YAAY,GAAG,CAAC,CAAC;IACzD,CAAC;IAED,MAAM,GAAG,GACP,WAAW;QACX,gBAAgB;QAChB,mBAAmB;QACnB,YAAY;QACZ,WAAW;QACX,YAAY;QACZ,gBAAgB,CAAC;IAEnB,IAAI,CAAC,CAAC,KAAK,KAAK,CAAC,IAAI,CAAC,CAAC,UAAU,KAAK,CAAC;QAAE,OAAO,CAAC,CAAC;IAElD,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;AACnC,CAAC;AAMD,SAAS,YAAY,CAAC,CAAe;IACnC,MAAM,OAAO,GAAa,EAAE,CAAC;IAC7B,IAAI,CAAC,CAAC,KAAK,IAAI,EAAE;QAAE,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,KAAK,WAAW,CAAC,CAAC;SAC9D,IAAI,CAAC,CAAC,KAAK,IAAI,CAAC;QAAE,OAAO,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,KAAK,WAAW,CAAC,CAAC;SACxE,IAAI,CAAC,CAAC,KAAK,GAAG,CAAC;QAAE,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,UAAU,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;IAEnF,IAAI,CAAC,CAAC,KAAK,IAAI,GAAG;QAAE,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC;SAC7D,IAAI,CAAC,CAAC,KAAK,IAAI,GAAG;QAAE,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,QAAQ,CAAC,CAAC;IAE1D,IAAI,CAAC,CAAC,OAAO,IAAI,CAAC;QAAE,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,eAAe,CAAC,CAAC;SACzD,IAAI,CAAC,CAAC,OAAO,IAAI,CAAC;QAAE,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,eAAe,CAAC,CAAC;IAEnE,IAAI,CAAC,CAAC,UAAU,GAAG,CAAC,EAAE,CAAC;QACrB,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,UAAU,cAAc,CAAC,CAAC,UAAU,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;IAC7E,CAAC;IAED,IAAI,CAAC,CAAC,mBAAmB,KAAK,IAAI,IAAI,CAAC,CAAC,mBAAmB,IAAI,CAAC,EAAE,CAAC;QACjE,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;IACpC,CAAC;IAED,IAAI,CAAC,CAAC,YAAY,IAAI,CAAC,CAAC,aAAa,EAAE,CAAC;QACtC,OAAO,CAAC,IAAI,CAAC,iBAAiB,YAAY,CAAC,CAAC,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;IAClE,CAAC;IAED,OAAO,OAAO,CAAC;AACjB,CAAC;AAED,SAAS,WAAW,CAAC,aAA8C;IACjE,IAAI,CAAC,aAAa,IAAI,aAAa,CAAC,IAAI,KAAK,CAAC;QAAE,OAAO,EAAE,CAAC;IAC1D,MAAM,KAAK,GAAG,CAAC,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;IACzE,IAAI,KAAK,KAAK,CAAC;QAAE,OAAO,EAAE,CAAC;IAE3B,OAAO,CAAC,GAAG,aAAa,CAAC,OAAO,EAAE,CAAC;SAChC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;SAC3B,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;SACX,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,EAAE,CAAC,CAAC;QAC3B,MAAM;QACN,OAAO;QACP,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,GAAG,KAAK,CAAC,GAAG,IAAI,CAAC,GAAG,IAAI;KACnD,CAAC,CAAC,CAAC;AACR,CAAC;AAED,SAAS,YAAY,CAAC,KAAa;IACjC,MAAM,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IACjC,OAAO,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;AACnD,CAAC"}
package/dist/index.d.ts CHANGED
@@ -3,5 +3,11 @@ export { detectLanguages } from './core/languageDetector.js';
3
3
  export { detectFrameworks } from './core/frameworkDetector.js';
4
4
  export { analyzeDependencies } from './core/dependencyAnalyzer.js';
5
5
  export { collectIssues } from './core/issueEngine.js';
6
+ export { analyzeHotspots, computeRiskScore } from './core/hotspotAnalyzer.js';
7
+ export { inspectFile } from './core/fileInspector.js';
6
8
  export { walkFiles } from './utils/fileWalker.js';
7
- export type { ScanResult, FileEntry, DirectoryNode, LanguageBreakdown, LanguageStat, FrameworkResult, DetectedFramework, DependencyReport, DependencyRisk, Issue, IssueSeverity, Fix, FixResult, FileExplanation, ImportInfo, ExportInfo, ArchitectureLayer, AnalysisReport, ReportFormat, } from './types.js';
9
+ export { createMcpServer, runMcpServer } from './mcp/server.js';
10
+ export { getToolDefinitions } from './mcp/tools.js';
11
+ export { getPromptDefinitions } from './mcp/prompts.js';
12
+ export { getResourceDefinitions } from './mcp/resources.js';
13
+ export type { ScanResult, FileEntry, DirectoryNode, LanguageBreakdown, LanguageStat, FrameworkResult, DetectedFramework, DependencyReport, DependencyRisk, Issue, IssueSeverity, Fix, FixResult, FileExplanation, FileInspection, ImportInfo, ExportInfo, ArchitectureLayer, AnalysisReport, ReportFormat, FileHotspot, HotspotReport, AuthorShare, BaselineHotspot, HotspotDelta, HotspotDiffSummary, McpToolDefinition, McpPromptDefinition, McpResourceDefinition, } from './types.js';
package/dist/index.js CHANGED
@@ -3,5 +3,11 @@ export { detectLanguages } from './core/languageDetector.js';
3
3
  export { detectFrameworks } from './core/frameworkDetector.js';
4
4
  export { analyzeDependencies } from './core/dependencyAnalyzer.js';
5
5
  export { collectIssues } from './core/issueEngine.js';
6
+ export { analyzeHotspots, computeRiskScore } from './core/hotspotAnalyzer.js';
7
+ export { inspectFile } from './core/fileInspector.js';
6
8
  export { walkFiles } from './utils/fileWalker.js';
9
+ export { createMcpServer, runMcpServer } from './mcp/server.js';
10
+ export { getToolDefinitions } from './mcp/tools.js';
11
+ export { getPromptDefinitions } from './mcp/prompts.js';
12
+ export { getResourceDefinitions } from './mcp/resources.js';
7
13
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,6BAA6B,CAAC;AAC7D,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC7D,OAAO,EAAE,gBAAgB,EAAE,MAAM,6BAA6B,CAAC;AAC/D,OAAO,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AACnE,OAAO,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,6BAA6B,CAAC;AAC7D,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC7D,OAAO,EAAE,gBAAgB,EAAE,MAAM,6BAA6B,CAAC;AAC/D,OAAO,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AACnE,OAAO,EAAE,aAAa,EAAE,MAAM,uBAAuB,CAAC;AACtD,OAAO,EAAE,eAAe,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC9E,OAAO,EAAE,WAAW,EAAE,MAAM,yBAAyB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC;AAClD,OAAO,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAChE,OAAO,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAC;AACpD,OAAO,EAAE,oBAAoB,EAAE,MAAM,kBAAkB,CAAC;AACxD,OAAO,EAAE,sBAAsB,EAAE,MAAM,oBAAoB,CAAC"}
@@ -0,0 +1,14 @@
1
+ import type { McpPromptDefinition } from '../types.js';
2
+ export interface McpPromptMessage {
3
+ role: 'user' | 'assistant' | 'system';
4
+ content: {
5
+ type: 'text';
6
+ text: string;
7
+ };
8
+ }
9
+ export interface McpPromptResult {
10
+ description: string;
11
+ messages: McpPromptMessage[];
12
+ }
13
+ export declare function getPromptDefinitions(): McpPromptDefinition[];
14
+ export declare function getPrompt(name: string, args: Record<string, unknown>, rootPath: string): Promise<McpPromptResult>;