semantic-complexity 0.0.7-741984fb → 0.0.13-2aaf183b

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 (163) hide show
  1. package/dist/__tests__/analyzers.test.d.ts +5 -0
  2. package/dist/__tests__/analyzers.test.d.ts.map +1 -0
  3. package/dist/__tests__/analyzers.test.js +357 -0
  4. package/dist/__tests__/analyzers.test.js.map +1 -0
  5. package/dist/__tests__/gate.test.d.ts +5 -0
  6. package/dist/__tests__/gate.test.d.ts.map +1 -0
  7. package/dist/__tests__/gate.test.js +140 -0
  8. package/dist/__tests__/gate.test.js.map +1 -0
  9. package/dist/analyzers/bread.d.ts +63 -0
  10. package/dist/analyzers/bread.d.ts.map +1 -0
  11. package/dist/analyzers/bread.js +415 -0
  12. package/dist/analyzers/bread.js.map +1 -0
  13. package/dist/analyzers/cheese.d.ts +111 -0
  14. package/dist/analyzers/cheese.d.ts.map +1 -0
  15. package/dist/analyzers/cheese.js +881 -0
  16. package/dist/analyzers/cheese.js.map +1 -0
  17. package/dist/analyzers/ham.d.ts +33 -0
  18. package/dist/analyzers/ham.d.ts.map +1 -0
  19. package/dist/analyzers/ham.js +264 -0
  20. package/dist/analyzers/ham.js.map +1 -0
  21. package/dist/analyzers/index.d.ts +7 -0
  22. package/dist/analyzers/index.d.ts.map +1 -0
  23. package/dist/analyzers/index.js +7 -0
  24. package/dist/analyzers/index.js.map +1 -0
  25. package/dist/budget/index.d.ts +50 -0
  26. package/dist/budget/index.d.ts.map +1 -0
  27. package/dist/budget/index.js +138 -0
  28. package/dist/budget/index.js.map +1 -0
  29. package/dist/cli/index.d.ts +6 -0
  30. package/dist/cli/index.d.ts.map +1 -0
  31. package/dist/cli/index.js +103 -0
  32. package/dist/cli/index.js.map +1 -0
  33. package/dist/gate/index.d.ts +31 -0
  34. package/dist/gate/index.d.ts.map +1 -0
  35. package/dist/gate/index.js +117 -0
  36. package/dist/gate/index.js.map +1 -0
  37. package/dist/gate/waiver.d.ts +54 -0
  38. package/dist/gate/waiver.d.ts.map +1 -0
  39. package/dist/gate/waiver.js +280 -0
  40. package/dist/gate/waiver.js.map +1 -0
  41. package/dist/index.d.ts +8 -34
  42. package/dist/index.d.ts.map +1 -1
  43. package/dist/index.js +8 -49
  44. package/dist/index.js.map +1 -1
  45. package/dist/mcp/index.d.ts +5 -0
  46. package/dist/mcp/index.d.ts.map +1 -0
  47. package/dist/mcp/index.js +140 -0
  48. package/dist/mcp/index.js.map +1 -0
  49. package/dist/recommend/index.d.ts +32 -0
  50. package/dist/recommend/index.d.ts.map +1 -0
  51. package/dist/recommend/index.js +169 -0
  52. package/dist/recommend/index.js.map +1 -0
  53. package/dist/simplex/index.d.ts +26 -0
  54. package/dist/simplex/index.d.ts.map +1 -0
  55. package/dist/simplex/index.js +100 -0
  56. package/dist/simplex/index.js.map +1 -0
  57. package/dist/types/index.d.ts +39 -0
  58. package/dist/types/index.d.ts.map +1 -0
  59. package/dist/types/index.js +5 -0
  60. package/dist/types/index.js.map +1 -0
  61. package/package.json +27 -52
  62. package/LICENSE +0 -21
  63. package/README.md +0 -72
  64. package/dist/ast/index.d.ts +0 -2
  65. package/dist/ast/index.d.ts.map +0 -1
  66. package/dist/ast/index.js +0 -2
  67. package/dist/ast/index.js.map +0 -1
  68. package/dist/ast/parser.d.ts +0 -27
  69. package/dist/ast/parser.d.ts.map +0 -1
  70. package/dist/ast/parser.js +0 -273
  71. package/dist/ast/parser.js.map +0 -1
  72. package/dist/canonical/convergence.d.ts +0 -56
  73. package/dist/canonical/convergence.d.ts.map +0 -1
  74. package/dist/canonical/convergence.js +0 -149
  75. package/dist/canonical/convergence.js.map +0 -1
  76. package/dist/canonical/index.d.ts +0 -11
  77. package/dist/canonical/index.d.ts.map +0 -1
  78. package/dist/canonical/index.js +0 -11
  79. package/dist/canonical/index.js.map +0 -1
  80. package/dist/canonical/profiles.d.ts +0 -40
  81. package/dist/canonical/profiles.d.ts.map +0 -1
  82. package/dist/canonical/profiles.js +0 -182
  83. package/dist/canonical/profiles.js.map +0 -1
  84. package/dist/canonical/types.d.ts +0 -124
  85. package/dist/canonical/types.d.ts.map +0 -1
  86. package/dist/canonical/types.js +0 -46
  87. package/dist/canonical/types.js.map +0 -1
  88. package/dist/compare.d.ts +0 -31
  89. package/dist/compare.d.ts.map +0 -1
  90. package/dist/compare.js +0 -354
  91. package/dist/compare.js.map +0 -1
  92. package/dist/context/index.d.ts +0 -41
  93. package/dist/context/index.d.ts.map +0 -1
  94. package/dist/context/index.js +0 -253
  95. package/dist/context/index.js.map +0 -1
  96. package/dist/gates/delta.d.ts +0 -45
  97. package/dist/gates/delta.d.ts.map +0 -1
  98. package/dist/gates/delta.js +0 -260
  99. package/dist/gates/delta.js.map +0 -1
  100. package/dist/gates/index.d.ts +0 -9
  101. package/dist/gates/index.d.ts.map +0 -1
  102. package/dist/gates/index.js +0 -9
  103. package/dist/gates/index.js.map +0 -1
  104. package/dist/gates/types.d.ts +0 -96
  105. package/dist/gates/types.d.ts.map +0 -1
  106. package/dist/gates/types.js +0 -59
  107. package/dist/gates/types.js.map +0 -1
  108. package/dist/graph/call.d.ts +0 -63
  109. package/dist/graph/call.d.ts.map +0 -1
  110. package/dist/graph/call.js +0 -240
  111. package/dist/graph/call.js.map +0 -1
  112. package/dist/graph/dependency.d.ts +0 -52
  113. package/dist/graph/dependency.d.ts.map +0 -1
  114. package/dist/graph/dependency.js +0 -296
  115. package/dist/graph/dependency.js.map +0 -1
  116. package/dist/graph/index.d.ts +0 -3
  117. package/dist/graph/index.d.ts.map +0 -1
  118. package/dist/graph/index.js +0 -3
  119. package/dist/graph/index.js.map +0 -1
  120. package/dist/metrics/cognitive.d.ts +0 -42
  121. package/dist/metrics/cognitive.d.ts.map +0 -1
  122. package/dist/metrics/cognitive.js +0 -204
  123. package/dist/metrics/cognitive.js.map +0 -1
  124. package/dist/metrics/cyclomatic.d.ts +0 -31
  125. package/dist/metrics/cyclomatic.d.ts.map +0 -1
  126. package/dist/metrics/cyclomatic.js +0 -121
  127. package/dist/metrics/cyclomatic.js.map +0 -1
  128. package/dist/metrics/dimensional.d.ts +0 -32
  129. package/dist/metrics/dimensional.d.ts.map +0 -1
  130. package/dist/metrics/dimensional.js +0 -560
  131. package/dist/metrics/dimensional.js.map +0 -1
  132. package/dist/metrics/index.d.ts +0 -26
  133. package/dist/metrics/index.d.ts.map +0 -1
  134. package/dist/metrics/index.js +0 -120
  135. package/dist/metrics/index.js.map +0 -1
  136. package/dist/metrics/meta.d.ts +0 -90
  137. package/dist/metrics/meta.d.ts.map +0 -1
  138. package/dist/metrics/meta.js +0 -144
  139. package/dist/metrics/meta.js.map +0 -1
  140. package/dist/tensor/canonical.d.ts +0 -53
  141. package/dist/tensor/canonical.d.ts.map +0 -1
  142. package/dist/tensor/canonical.js +0 -192
  143. package/dist/tensor/canonical.js.map +0 -1
  144. package/dist/tensor/index.d.ts +0 -22
  145. package/dist/tensor/index.d.ts.map +0 -1
  146. package/dist/tensor/index.js +0 -22
  147. package/dist/tensor/index.js.map +0 -1
  148. package/dist/tensor/matrix.d.ts +0 -62
  149. package/dist/tensor/matrix.d.ts.map +0 -1
  150. package/dist/tensor/matrix.js +0 -187
  151. package/dist/tensor/matrix.js.map +0 -1
  152. package/dist/tensor/scoring.d.ts +0 -82
  153. package/dist/tensor/scoring.d.ts.map +0 -1
  154. package/dist/tensor/scoring.js +0 -233
  155. package/dist/tensor/scoring.js.map +0 -1
  156. package/dist/tensor/types.d.ts +0 -186
  157. package/dist/tensor/types.d.ts.map +0 -1
  158. package/dist/tensor/types.js +0 -15
  159. package/dist/tensor/types.js.map +0 -1
  160. package/dist/types.d.ts +0 -341
  161. package/dist/types.d.ts.map +0 -1
  162. package/dist/types.js +0 -22
  163. package/dist/types.js.map +0 -1
@@ -1,253 +0,0 @@
1
- import * as fs from 'node:fs';
2
- import * as path from 'node:path';
3
- import { parseSourceFile } from '../ast/parser.js';
4
- import { analyzeFile } from '../metrics/index.js';
5
- import { DependencyGraph } from '../graph/dependency.js';
6
- import { CallGraph } from '../graph/call.js';
7
- /**
8
- * MCP용 분석 컨텍스트 수집기
9
- *
10
- * LLM이 인지복잡도를 정확히 판단할 수 있도록
11
- * 코드와 관련된 모든 컨텍스트를 수집합니다.
12
- */
13
- export class ContextCollector {
14
- options;
15
- projectRoot;
16
- dependencyGraph;
17
- projectInfo;
18
- constructor(projectRoot, options) {
19
- this.projectRoot = path.resolve(projectRoot);
20
- this.options = { ...getDefaultOptions(), ...options };
21
- this.dependencyGraph = new DependencyGraph(this.projectRoot);
22
- this.projectInfo = this.detectProjectInfo();
23
- }
24
- /**
25
- * 파일에 대한 전체 분석 컨텍스트 수집
26
- */
27
- async collectContext(filePath) {
28
- const absPath = path.resolve(filePath);
29
- const content = fs.readFileSync(absPath, 'utf-8');
30
- const sourceFile = parseSourceFile(absPath, content);
31
- // 기본 분석
32
- const fileResult = analyzeFile(sourceFile);
33
- // 의존성 분석
34
- const depNode = this.dependencyGraph.analyzeFile(absPath, content);
35
- // 호출 그래프
36
- const callGraph = new CallGraph();
37
- callGraph.analyzeSourceFile(sourceFile);
38
- // 관련 파일 분석
39
- const relatedFiles = await this.analyzeRelatedFiles(absPath, this.options.contextDepth);
40
- return {
41
- file: fileResult,
42
- dependencies: depNode,
43
- callGraph: callGraph.getAllNodes(),
44
- relatedFiles,
45
- projectInfo: this.projectInfo,
46
- };
47
- }
48
- /**
49
- * MCP 응답용 프롬프트 생성
50
- *
51
- * LLM이 인지복잡도를 판단할 수 있도록
52
- * 구조화된 프롬프트를 생성합니다.
53
- */
54
- generatePrompt(context, functionName) {
55
- const lines = [];
56
- // 헤더
57
- lines.push('# 코드 복잡도 분석 요청');
58
- lines.push('');
59
- // 프로젝트 정보
60
- if (context.projectInfo) {
61
- lines.push('## 프로젝트 정보');
62
- lines.push(`- 이름: ${context.projectInfo.name}`);
63
- lines.push(`- 언어: ${context.projectInfo.language}`);
64
- if (context.projectInfo.framework) {
65
- lines.push(`- 프레임워크: ${context.projectInfo.framework}`);
66
- }
67
- lines.push('');
68
- }
69
- // 파일 정보
70
- lines.push('## 분석 대상 파일');
71
- lines.push(`경로: ${context.file.filePath}`);
72
- lines.push('');
73
- // 기계적 메트릭
74
- lines.push('## 기계적 복잡도 (참고용)');
75
- const targetFunctions = functionName
76
- ? context.file.functions.filter((f) => f.function.name === functionName)
77
- : context.file.functions;
78
- for (const func of targetFunctions) {
79
- lines.push(`### ${func.function.name}`);
80
- lines.push(`- 위치: ${func.function.location.startLine}행`);
81
- lines.push(`- 순환복잡도: ${func.cyclomatic}`);
82
- lines.push(`- 인지복잡도 (SonarSource): ${func.cognitive}`);
83
- lines.push('');
84
- if (func.details.length > 0) {
85
- lines.push('상세:');
86
- for (const detail of func.details.slice(0, 10)) {
87
- lines.push(` - ${detail.description} (${detail.location.startLine}행)`);
88
- }
89
- if (func.details.length > 10) {
90
- lines.push(` - ... 외 ${func.details.length - 10}개`);
91
- }
92
- lines.push('');
93
- }
94
- }
95
- // 의존성 정보
96
- lines.push('## 의존성 관계');
97
- lines.push(`이 파일이 import하는 모듈: ${context.dependencies.imports.length}개`);
98
- for (const imp of context.dependencies.imports.slice(0, 5)) {
99
- lines.push(` - ${path.basename(imp)}`);
100
- }
101
- lines.push(`이 파일을 import하는 모듈: ${context.dependencies.importedBy.length}개`);
102
- for (const dep of context.dependencies.importedBy.slice(0, 5)) {
103
- lines.push(` - ${path.basename(dep)}`);
104
- }
105
- lines.push('');
106
- // 호출 관계
107
- if (context.callGraph.length > 0) {
108
- lines.push('## 함수 호출 관계');
109
- for (const node of context.callGraph.slice(0, 5)) {
110
- const callCount = node.calls.length;
111
- const calledByCount = node.calledBy.length;
112
- lines.push(`- ${node.function.name}: 호출 ${callCount}개, 호출됨 ${calledByCount}개`);
113
- }
114
- lines.push('');
115
- }
116
- // 분석 요청
117
- lines.push('## 분석 요청');
118
- lines.push('위 정보를 바탕으로 다음을 분석해주세요:');
119
- lines.push('');
120
- lines.push('1. **실제 인지복잡도 평가**');
121
- lines.push(' - 알고리즘 패턴 인식 (매핑, 변환, 상태 머신 등)');
122
- lines.push(' - 도메인 로직 난이도');
123
- lines.push(' - 추상화 수준 적절성');
124
- lines.push('');
125
- lines.push('2. **리팩토링 제안** (필요시)');
126
- lines.push(' - 함수 분리 가능 지점');
127
- lines.push(' - 패턴 적용 가능성');
128
- lines.push(' - 테스트 용이성 개선 방안');
129
- lines.push('');
130
- lines.push('3. **복잡도 점수 (1-100)**');
131
- lines.push(' - 기계적 메트릭과 실제 복잡도의 차이 설명');
132
- return lines.join('\n');
133
- }
134
- /**
135
- * 관련 파일 분석
136
- */
137
- async analyzeRelatedFiles(filePath, depth) {
138
- const results = [];
139
- const depTree = this.dependencyGraph.buildDependencyTree(filePath, depth);
140
- for (const node of depTree) {
141
- if (node.filePath === filePath)
142
- continue;
143
- try {
144
- const content = fs.readFileSync(node.filePath, 'utf-8');
145
- const sourceFile = parseSourceFile(node.filePath, content);
146
- const result = analyzeFile(sourceFile);
147
- results.push(result);
148
- }
149
- catch {
150
- // 파일 읽기 실패 무시
151
- }
152
- }
153
- return results;
154
- }
155
- /**
156
- * 프로젝트 정보 감지
157
- */
158
- detectProjectInfo() {
159
- try {
160
- const packageJsonPath = path.join(this.projectRoot, 'package.json');
161
- if (!fs.existsSync(packageJsonPath))
162
- return undefined;
163
- const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
164
- // 패키지 매니저 감지
165
- let packageManager = 'npm';
166
- if (fs.existsSync(path.join(this.projectRoot, 'pnpm-lock.yaml'))) {
167
- packageManager = 'pnpm';
168
- }
169
- else if (fs.existsSync(path.join(this.projectRoot, 'yarn.lock'))) {
170
- packageManager = 'yarn';
171
- }
172
- else if (fs.existsSync(path.join(this.projectRoot, 'bun.lockb'))) {
173
- packageManager = 'bun';
174
- }
175
- // 프레임워크 감지
176
- const deps = {
177
- ...packageJson.dependencies,
178
- ...packageJson.devDependencies,
179
- };
180
- let framework;
181
- if (deps['next'])
182
- framework = 'Next.js';
183
- else if (deps['nuxt'])
184
- framework = 'Nuxt';
185
- else if (deps['@angular/core'])
186
- framework = 'Angular';
187
- else if (deps['vue'])
188
- framework = 'Vue';
189
- else if (deps['react'])
190
- framework = 'React';
191
- else if (deps['svelte'])
192
- framework = 'Svelte';
193
- else if (deps['express'])
194
- framework = 'Express';
195
- else if (deps['fastify'])
196
- framework = 'Fastify';
197
- else if (deps['nestjs'])
198
- framework = 'NestJS';
199
- // 언어 감지
200
- const hasTsConfig = fs.existsSync(path.join(this.projectRoot, 'tsconfig.json'));
201
- const language = hasTsConfig
202
- ? 'typescript'
203
- : 'javascript';
204
- return {
205
- name: packageJson.name || path.basename(this.projectRoot),
206
- root: this.projectRoot,
207
- packageManager,
208
- framework,
209
- language,
210
- };
211
- }
212
- catch {
213
- return undefined;
214
- }
215
- }
216
- }
217
- /**
218
- * 기본 옵션 반환
219
- */
220
- function getDefaultOptions() {
221
- return {
222
- cyclomaticThreshold: 10,
223
- cognitiveThreshold: 15,
224
- includeTypes: true,
225
- excludeTests: true,
226
- excludePatterns: ['node_modules', 'dist', 'build', '.git'],
227
- contextDepth: 2,
228
- };
229
- }
230
- /**
231
- * 단일 파일 빠른 분석 (MCP 도구용)
232
- */
233
- export async function quickAnalyze(filePath, functionName) {
234
- const projectRoot = findProjectRoot(filePath);
235
- const collector = new ContextCollector(projectRoot);
236
- const context = await collector.collectContext(filePath);
237
- const prompt = collector.generatePrompt(context, functionName);
238
- return { context, prompt };
239
- }
240
- /**
241
- * 프로젝트 루트 찾기
242
- */
243
- function findProjectRoot(startPath) {
244
- let current = path.dirname(path.resolve(startPath));
245
- while (current !== path.dirname(current)) {
246
- if (fs.existsSync(path.join(current, 'package.json'))) {
247
- return current;
248
- }
249
- current = path.dirname(current);
250
- }
251
- return path.dirname(startPath);
252
- }
253
- //# sourceMappingURL=index.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/context/index.ts"],"names":[],"mappings":"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"}
@@ -1,45 +0,0 @@
1
- /**
2
- * Delta 분석 (Δ Analysis)
3
- *
4
- * "나쁜 코드"가 아닌 "나빠지는 변경"을 감지
5
- *
6
- * 변경량 기반 품질 게이트:
7
- * - baseline과 current 비교
8
- * - 임계값 초과 시 위반 생성
9
- * - 담당 게이트 할당
10
- */
11
- import type { MetaDimensions, Snapshot, ModuleType } from '../canonical/types.js';
12
- import type { DeltaAnalysis, DeltaThresholds, Violation, GateType, GateResult, GatePipelineResult } from './types.js';
13
- /**
14
- * Delta 계산 (current - baseline)
15
- */
16
- export declare function calculateDelta(baseline: MetaDimensions, current: MetaDimensions): MetaDimensions;
17
- /**
18
- * Delta 퍼센트 계산
19
- */
20
- export declare function calculateDeltaPercent(baseline: MetaDimensions, delta: MetaDimensions): MetaDimensions;
21
- /**
22
- * 위반 검출
23
- */
24
- export declare function detectViolations(baseline: MetaDimensions, current: MetaDimensions, thresholds?: DeltaThresholds): Violation[];
25
- /**
26
- * Delta 분석 수행
27
- */
28
- export declare function analyzeDelta(baseline: Snapshot, current: Snapshot, thresholds?: DeltaThresholds): DeltaAnalysis;
29
- /**
30
- * 단일 게이트 검사
31
- */
32
- export declare function checkGate(gate: GateType, violations: Violation[]): GateResult;
33
- /**
34
- * 전체 게이트 파이프라인 실행
35
- */
36
- export declare function runGatePipeline(moduleType: ModuleType, baseline: Snapshot, current: Snapshot, thresholds?: DeltaThresholds): GatePipelineResult;
37
- /**
38
- * 스냅샷 생성 헬퍼
39
- */
40
- export declare function createSnapshot(moduleType: ModuleType, meta: MetaDimensions, options?: {
41
- commitHash?: string;
42
- functionCount?: number;
43
- totalWeighted?: number;
44
- }): Snapshot;
45
- //# sourceMappingURL=delta.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"delta.d.ts","sourceRoot":"","sources":["../../src/gates/delta.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,KAAK,EAAE,cAAc,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAC;AAClF,OAAO,KAAK,EACV,aAAa,EACb,eAAe,EACf,SAAS,EAET,QAAQ,EACR,UAAU,EAEV,kBAAkB,EACnB,MAAM,YAAY,CAAC;AAYpB;;GAEG;AACH,wBAAgB,cAAc,CAC5B,QAAQ,EAAE,cAAc,EACxB,OAAO,EAAE,cAAc,GACtB,cAAc,CAEhB;AAED;;GAEG;AACH,wBAAgB,qBAAqB,CACnC,QAAQ,EAAE,cAAc,EACxB,KAAK,EAAE,cAAc,GACpB,cAAc,CAShB;AAyCD;;GAEG;AACH,wBAAgB,gBAAgB,CAC9B,QAAQ,EAAE,cAAc,EACxB,OAAO,EAAE,cAAc,EACvB,UAAU,GAAE,eAA0C,GACrD,SAAS,EAAE,CAoCb;AA8DD;;GAEG;AACH,wBAAgB,YAAY,CAC1B,QAAQ,EAAE,QAAQ,EAClB,OAAO,EAAE,QAAQ,EACjB,UAAU,GAAE,eAA0C,GACrD,aAAa,CAmBf;AAMD;;GAEG;AACH,wBAAgB,SAAS,CACvB,IAAI,EAAE,QAAQ,EACd,UAAU,EAAE,SAAS,EAAE,GACtB,UAAU,CA0BZ;AAED;;GAEG;AACH,wBAAgB,eAAe,CAC7B,UAAU,EAAE,UAAU,EACtB,QAAQ,EAAE,QAAQ,EAClB,OAAO,EAAE,QAAQ,EACjB,UAAU,GAAE,eAA0C,GACrD,kBAAkB,CAqCpB;AA2BD;;GAEG;AACH,wBAAgB,cAAc,CAC5B,UAAU,EAAE,UAAU,EACtB,IAAI,EAAE,cAAc,EACpB,OAAO,GAAE;IACP,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,aAAa,CAAC,EAAE,MAAM,CAAC;CACnB,GACL,QAAQ,CASV"}
@@ -1,260 +0,0 @@
1
- /**
2
- * Delta 분석 (Δ Analysis)
3
- *
4
- * "나쁜 코드"가 아닌 "나빠지는 변경"을 감지
5
- *
6
- * 변경량 기반 품질 게이트:
7
- * - baseline과 current 비교
8
- * - 임계값 초과 시 위반 생성
9
- * - 담당 게이트 할당
10
- */
11
- import { DEFAULT_DELTA_THRESHOLDS, GATE_RESPONSIBILITIES, GATE_INFO, } from './types.js';
12
- import { subtractMetaDimensions } from '../metrics/meta.js';
13
- // ─────────────────────────────────────────────────────────────────
14
- // Delta 계산
15
- // ─────────────────────────────────────────────────────────────────
16
- /**
17
- * Delta 계산 (current - baseline)
18
- */
19
- export function calculateDelta(baseline, current) {
20
- return subtractMetaDimensions(current, baseline);
21
- }
22
- /**
23
- * Delta 퍼센트 계산
24
- */
25
- export function calculateDeltaPercent(baseline, delta) {
26
- const safeDiv = (d, b) => b === 0 ? (d > 0 ? 100 : 0) : Math.round((d / b) * 100 * 10) / 10;
27
- return {
28
- security: safeDiv(delta.security, baseline.security),
29
- context: safeDiv(delta.context, baseline.context),
30
- behavior: safeDiv(delta.behavior, baseline.behavior),
31
- };
32
- }
33
- // ─────────────────────────────────────────────────────────────────
34
- // 위반 검출
35
- // ─────────────────────────────────────────────────────────────────
36
- /**
37
- * 심각도 결정
38
- */
39
- function determineSeverity(delta, threshold, deltaPercent) {
40
- const ratio = delta / threshold;
41
- const percentRatio = deltaPercent / 100;
42
- if (ratio >= 2 || percentRatio >= 1) {
43
- return 'critical';
44
- }
45
- if (ratio >= 1.5 || percentRatio >= 0.5) {
46
- return 'error';
47
- }
48
- if (ratio >= 1 || percentRatio >= 0.2) {
49
- return 'warning';
50
- }
51
- return 'info';
52
- }
53
- /**
54
- * 차원별 게이트 찾기
55
- */
56
- function findResponsibleGate(dimension) {
57
- for (const [gate, dim] of Object.entries(GATE_RESPONSIBILITIES)) {
58
- if (dim === dimension) {
59
- return gate;
60
- }
61
- }
62
- return 'dev'; // fallback
63
- }
64
- /**
65
- * 위반 검출
66
- */
67
- export function detectViolations(baseline, current, thresholds = DEFAULT_DELTA_THRESHOLDS) {
68
- const violations = [];
69
- const delta = calculateDelta(baseline, current);
70
- const deltaPercent = calculateDeltaPercent(baseline, delta);
71
- const dimensions = ['security', 'context', 'behavior'];
72
- for (const dim of dimensions) {
73
- const d = delta[dim];
74
- const dp = deltaPercent[dim];
75
- const absThreshold = thresholds.absolute[dim];
76
- const relThreshold = thresholds.relative[dim];
77
- // 증가만 감지 (감소는 개선이므로 무시)
78
- if (d <= 0)
79
- continue;
80
- // 절대값 또는 상대값 임계 초과
81
- if (d > absThreshold || dp > relThreshold) {
82
- const gate = findResponsibleGate(dim);
83
- const severity = determineSeverity(d, absThreshold, dp);
84
- violations.push({
85
- gate,
86
- dimension: dim,
87
- severity,
88
- currentValue: current[dim],
89
- previousValue: baseline[dim],
90
- delta: Math.round(d * 100) / 100,
91
- threshold: absThreshold,
92
- message: generateViolationMessage(dim, d, dp, severity),
93
- suggestion: generateSuggestion(dim, severity),
94
- });
95
- }
96
- }
97
- return violations;
98
- }
99
- /**
100
- * 위반 메시지 생성
101
- */
102
- function generateViolationMessage(dimension, delta, deltaPercent, severity) {
103
- const dimNames = {
104
- security: '🍞 Security (구조 안정성)',
105
- context: '🧀 Context (맥락 밀도)',
106
- behavior: '🥓 Behavior (행동 보존성)',
107
- };
108
- const severityLabels = {
109
- info: 'ℹ️',
110
- warning: '⚠️',
111
- error: '❌',
112
- critical: '🚨',
113
- };
114
- return `${severityLabels[severity]} ${dimNames[dimension]}: +${delta.toFixed(1)} (+${deltaPercent.toFixed(1)}%)`;
115
- }
116
- /**
117
- * 개선 제안 생성
118
- */
119
- function generateSuggestion(dimension, severity) {
120
- const suggestions = {
121
- security: {
122
- info: '외부 의존성 변경을 확인하세요.',
123
- warning: '전역 변수 접근을 줄이세요.',
124
- error: '환경 의존성을 격리하세요.',
125
- critical: '보안 검토가 필수입니다. RA 승인을 받으세요.',
126
- },
127
- context: {
128
- info: '코드 복잡도가 증가했습니다.',
129
- warning: '중첩 깊이를 줄이세요.',
130
- error: '함수를 분리하세요.',
131
- critical: '리팩토링이 필수입니다. 설계 검토를 진행하세요.',
132
- },
133
- behavior: {
134
- info: '상태 변경이 추가되었습니다.',
135
- warning: '부작용을 최소화하세요.',
136
- error: '상태 관리를 단순화하세요.',
137
- critical: '행동 변경이 큽니다. 전체 테스트가 필수입니다.',
138
- },
139
- };
140
- return suggestions[dimension][severity];
141
- }
142
- // ─────────────────────────────────────────────────────────────────
143
- // Delta 분석
144
- // ─────────────────────────────────────────────────────────────────
145
- /**
146
- * Delta 분석 수행
147
- */
148
- export function analyzeDelta(baseline, current, thresholds = DEFAULT_DELTA_THRESHOLDS) {
149
- const delta = calculateDelta(baseline.meta, current.meta);
150
- const deltaPercent = calculateDeltaPercent(baseline.meta, delta);
151
- const violations = detectViolations(baseline.meta, current.meta, thresholds);
152
- const affectedGates = [...new Set(violations.map((v) => v.gate))];
153
- const exceedsThreshold = violations.some((v) => v.severity === 'error' || v.severity === 'critical');
154
- return {
155
- baseline,
156
- current,
157
- delta,
158
- deltaPercent,
159
- exceedsThreshold,
160
- violations,
161
- affectedGates,
162
- };
163
- }
164
- // ─────────────────────────────────────────────────────────────────
165
- // 게이트 파이프라인
166
- // ─────────────────────────────────────────────────────────────────
167
- /**
168
- * 단일 게이트 검사
169
- */
170
- export function checkGate(gate, violations) {
171
- const gateViolations = violations.filter((v) => v.gate === gate);
172
- let decision = 'pass';
173
- if (gateViolations.some((v) => v.severity === 'critical')) {
174
- decision = 'fail';
175
- }
176
- else if (gateViolations.some((v) => v.severity === 'error')) {
177
- decision = 'fail';
178
- }
179
- else if (gateViolations.some((v) => v.severity === 'warning')) {
180
- decision = 'warn';
181
- }
182
- const info = GATE_INFO[gate];
183
- const summary = decision === 'pass'
184
- ? `${info.name}: 통과`
185
- : decision === 'warn'
186
- ? `${info.name}: 주의 필요 (${gateViolations.length}건)`
187
- : `${info.name}: 승인 필요 (${gateViolations.length}건)`;
188
- return {
189
- gate,
190
- decision,
191
- violations: gateViolations,
192
- summary,
193
- };
194
- }
195
- /**
196
- * 전체 게이트 파이프라인 실행
197
- */
198
- export function runGatePipeline(moduleType, baseline, current, thresholds = DEFAULT_DELTA_THRESHOLDS) {
199
- const deltaResult = analyzeDelta(baseline, current, thresholds);
200
- const gates = ['dev', 'qa', 'ra'];
201
- const gateResults = gates.map((gate) => checkGate(gate, deltaResult.violations));
202
- // 전체 결정
203
- let overallDecision = 'pass';
204
- if (gateResults.some((g) => g.decision === 'fail')) {
205
- overallDecision = 'fail';
206
- }
207
- else if (gateResults.some((g) => g.decision === 'warn')) {
208
- overallDecision = 'warn';
209
- }
210
- // 필요한 승인
211
- const requiredApprovals = gateResults
212
- .filter((g) => g.decision === 'fail')
213
- .map((g) => g.gate);
214
- // CEO 승인 필요 여부 (critical 위반 시)
215
- if (deltaResult.violations.some((v) => v.severity === 'critical')) {
216
- requiredApprovals.push('ceo');
217
- }
218
- const summary = generatePipelineSummary(overallDecision, requiredApprovals);
219
- return {
220
- moduleType,
221
- baseline,
222
- current,
223
- gates: gateResults,
224
- overallDecision,
225
- requiredApprovals: [...new Set(requiredApprovals)],
226
- summary,
227
- };
228
- }
229
- /**
230
- * 파이프라인 요약 생성
231
- */
232
- function generatePipelineSummary(decision, requiredApprovals) {
233
- if (decision === 'pass') {
234
- return '✅ 모든 게이트 통과. 자동 머지 가능.';
235
- }
236
- if (decision === 'warn') {
237
- return '⚠️ 주의 필요. 리뷰 후 머지 권장.';
238
- }
239
- const approvers = requiredApprovals
240
- .map((g) => GATE_INFO[g].name)
241
- .join(', ');
242
- return `❌ 승인 필요: ${approvers}`;
243
- }
244
- // ─────────────────────────────────────────────────────────────────
245
- // 유틸리티
246
- // ─────────────────────────────────────────────────────────────────
247
- /**
248
- * 스냅샷 생성 헬퍼
249
- */
250
- export function createSnapshot(moduleType, meta, options = {}) {
251
- return {
252
- timestamp: Date.now(),
253
- commitHash: options.commitHash,
254
- moduleType,
255
- meta,
256
- functionCount: options.functionCount ?? 0,
257
- totalWeighted: options.totalWeighted ?? 0,
258
- };
259
- }
260
- //# sourceMappingURL=delta.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"delta.js","sourceRoot":"","sources":["../../src/gates/delta.ts"],"names":[],"mappings":"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"}
@@ -1,9 +0,0 @@
1
- /**
2
- * 게이트 시스템
3
- *
4
- * Delta 기반 품질 게이트 및 다단계 승인 워크플로우
5
- */
6
- export type { GateType, GateInfo, ViolationSeverity, Violation, DeltaAnalysis, DeltaThresholds, GateDecision, GateResult, GatePipelineResult, } from './types.js';
7
- export { GATE_RESPONSIBILITIES, GATE_INFO, DEFAULT_DELTA_THRESHOLDS, } from './types.js';
8
- export { calculateDelta, calculateDeltaPercent, detectViolations, analyzeDelta, checkGate, runGatePipeline, createSnapshot, } from './delta.js';
9
- //# sourceMappingURL=index.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/gates/index.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAGH,YAAY,EACV,QAAQ,EACR,QAAQ,EACR,iBAAiB,EACjB,SAAS,EACT,aAAa,EACb,eAAe,EACf,YAAY,EACZ,UAAU,EACV,kBAAkB,GACnB,MAAM,YAAY,CAAC;AAEpB,OAAO,EACL,qBAAqB,EACrB,SAAS,EACT,wBAAwB,GACzB,MAAM,YAAY,CAAC;AAGpB,OAAO,EACL,cAAc,EACd,qBAAqB,EACrB,gBAAgB,EAChB,YAAY,EACZ,SAAS,EACT,eAAe,EACf,cAAc,GACf,MAAM,YAAY,CAAC"}
@@ -1,9 +0,0 @@
1
- /**
2
- * 게이트 시스템
3
- *
4
- * Delta 기반 품질 게이트 및 다단계 승인 워크플로우
5
- */
6
- export { GATE_RESPONSIBILITIES, GATE_INFO, DEFAULT_DELTA_THRESHOLDS, } from './types.js';
7
- // Delta Analysis
8
- export { calculateDelta, calculateDeltaPercent, detectViolations, analyzeDelta, checkGate, runGatePipeline, createSnapshot, } from './delta.js';
9
- //# sourceMappingURL=index.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/gates/index.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAeH,OAAO,EACL,qBAAqB,EACrB,SAAS,EACT,wBAAwB,GACzB,MAAM,YAAY,CAAC;AAEpB,iBAAiB;AACjB,OAAO,EACL,cAAc,EACd,qBAAqB,EACrB,gBAAgB,EAChB,YAAY,EACZ,SAAS,EACT,eAAe,EACf,cAAc,GACf,MAAM,YAAY,CAAC"}