@girardelli/architect 5.0.0 → 8.1.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.
- package/dist/{cli.d.ts → src/adapters/cli.d.ts} +1 -2
- package/dist/{cli.js → src/adapters/cli.js} +191 -213
- package/dist/src/adapters/cli.js.map +1 -0
- package/dist/src/adapters/github-action.d.ts +9 -0
- package/dist/src/adapters/github-action.js +94 -0
- package/dist/src/adapters/github-action.js.map +1 -0
- package/dist/src/adapters/html-reporter/scripts.d.ts +5 -0
- package/dist/src/adapters/html-reporter/scripts.js +400 -0
- package/dist/src/adapters/html-reporter/scripts.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/agents.d.ts +2 -0
- package/dist/src/adapters/html-reporter/sections/agents.js +260 -0
- package/dist/src/adapters/html-reporter/sections/agents.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/anti-patterns.d.ts +13 -0
- package/dist/src/adapters/html-reporter/sections/anti-patterns.js +64 -0
- package/dist/src/adapters/html-reporter/sections/anti-patterns.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/header.d.ts +3 -0
- package/dist/src/adapters/html-reporter/sections/header.js +30 -0
- package/dist/src/adapters/html-reporter/sections/header.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/layers.d.ts +9 -0
- package/dist/src/adapters/html-reporter/sections/layers.js +143 -0
- package/dist/src/adapters/html-reporter/sections/layers.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/overview.d.ts +2 -0
- package/dist/src/adapters/html-reporter/sections/overview.js +58 -0
- package/dist/src/adapters/html-reporter/sections/overview.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/refactoring-plan.d.ts +3 -0
- package/dist/src/adapters/html-reporter/sections/refactoring-plan.js +151 -0
- package/dist/src/adapters/html-reporter/sections/refactoring-plan.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/score.d.ts +7 -0
- package/dist/src/adapters/html-reporter/sections/score.js +70 -0
- package/dist/src/adapters/html-reporter/sections/score.js.map +1 -0
- package/dist/src/adapters/html-reporter/sections/suggestions.d.ts +7 -0
- package/dist/src/adapters/html-reporter/sections/suggestions.js +34 -0
- package/dist/src/adapters/html-reporter/sections/suggestions.js.map +1 -0
- package/dist/src/adapters/html-reporter/styles.d.ts +1 -0
- package/dist/src/adapters/html-reporter/styles.js +526 -0
- package/dist/src/adapters/html-reporter/styles.js.map +1 -0
- package/dist/src/adapters/html-reporter/utils_adapters.d.ts +20 -0
- package/dist/src/adapters/html-reporter/utils_adapters.js +32 -0
- package/dist/src/adapters/html-reporter/utils_adapters.js.map +1 -0
- package/dist/src/adapters/html-reporter/utils_sections.d.ts +7 -0
- package/dist/src/adapters/html-reporter/utils_sections.js +58 -0
- package/dist/src/adapters/html-reporter/utils_sections.js.map +1 -0
- package/dist/src/adapters/html-reporter.d.ts +10 -0
- package/dist/src/adapters/html-reporter.js +97 -0
- package/dist/src/adapters/html-reporter.js.map +1 -0
- package/dist/src/adapters/progress-logger.d.ts +55 -0
- package/dist/src/adapters/progress-logger.js +200 -0
- package/dist/src/adapters/progress-logger.js.map +1 -0
- package/dist/{refactor-reporter.d.ts → src/adapters/refactor-reporter.d.ts} +1 -2
- package/dist/{refactor-reporter.js → src/adapters/refactor-reporter.js} +1 -1
- package/dist/src/adapters/refactor-reporter.js.map +1 -0
- package/dist/{reporter.d.ts → src/adapters/reporter.d.ts} +1 -2
- package/dist/src/adapters/reporter.js.map +1 -0
- package/dist/src/core/GenesisTerminal.d.ts +8 -0
- package/dist/src/core/GenesisTerminal.js +105 -0
- package/dist/src/core/GenesisTerminal.js.map +1 -0
- package/dist/{index.d.ts → src/core/architect.d.ts} +4 -18
- package/dist/{index.js → src/core/architect.js} +22 -21
- package/dist/src/core/architect.js.map +1 -0
- package/dist/tests/architect-adapter-enrichment.test.d.ts +1 -0
- package/dist/tests/architect-adapter-enrichment.test.js +11 -0
- package/dist/tests/architect-adapter-enrichment.test.js.map +1 -0
- package/dist/tests/github-action.test.d.ts +1 -0
- package/dist/tests/github-action.test.js +92 -0
- package/dist/tests/github-action.test.js.map +1 -0
- package/package.json +15 -65
- package/src/adapters/cli.ts +492 -0
- package/src/adapters/github-action.ts +109 -0
- package/src/adapters/html-reporter/scripts.ts +402 -0
- package/src/adapters/html-reporter/sections/agents.ts +267 -0
- package/src/adapters/html-reporter/sections/anti-patterns.ts +81 -0
- package/src/adapters/html-reporter/sections/header.ts +35 -0
- package/src/adapters/html-reporter/sections/layers.ts +165 -0
- package/src/adapters/html-reporter/sections/overview.ts +64 -0
- package/src/adapters/html-reporter/sections/refactoring-plan.ts +166 -0
- package/src/adapters/html-reporter/sections/score.ts +80 -0
- package/src/adapters/html-reporter/sections/suggestions.ts +39 -0
- package/src/adapters/html-reporter/styles.ts +525 -0
- package/src/adapters/html-reporter/utils_adapters.ts +39 -0
- package/src/adapters/html-reporter/utils_sections.ts +55 -0
- package/src/adapters/html-reporter.ts +102 -0
- package/src/adapters/progress-logger.ts +236 -0
- package/src/{refactor-reporter.ts → adapters/refactor-reporter.ts} +2 -2
- package/src/{reporter.ts → adapters/reporter.ts} +1 -1
- package/src/core/GenesisTerminal.ts +127 -0
- package/src/{index.ts → core/architect.ts} +27 -45
- package/tests/github-action.test.ts +109 -0
- package/tsconfig.json +12 -19
- package/CONTRIBUTING.md +0 -140
- package/LICENSE +0 -21
- package/PROJECT_STRUCTURE.txt +0 -168
- package/README.md +0 -257
- package/architect-run.sh +0 -431
- package/assets/banner-v3.html +0 -561
- package/dist/agent-generator/context-enricher.d.ts +0 -58
- package/dist/agent-generator/context-enricher.d.ts.map +0 -1
- package/dist/agent-generator/context-enricher.js +0 -613
- package/dist/agent-generator/context-enricher.js.map +0 -1
- package/dist/agent-generator/domain-inferrer.d.ts +0 -52
- package/dist/agent-generator/domain-inferrer.d.ts.map +0 -1
- package/dist/agent-generator/domain-inferrer.js +0 -585
- package/dist/agent-generator/domain-inferrer.js.map +0 -1
- package/dist/agent-generator/framework-detector.d.ts +0 -40
- package/dist/agent-generator/framework-detector.d.ts.map +0 -1
- package/dist/agent-generator/framework-detector.js +0 -611
- package/dist/agent-generator/framework-detector.js.map +0 -1
- package/dist/agent-generator/index.d.ts +0 -47
- package/dist/agent-generator/index.d.ts.map +0 -1
- package/dist/agent-generator/index.js +0 -545
- package/dist/agent-generator/index.js.map +0 -1
- package/dist/agent-generator/stack-detector.d.ts +0 -14
- package/dist/agent-generator/stack-detector.d.ts.map +0 -1
- package/dist/agent-generator/stack-detector.js +0 -124
- package/dist/agent-generator/stack-detector.js.map +0 -1
- package/dist/agent-generator/templates/core/agents.d.ts +0 -17
- package/dist/agent-generator/templates/core/agents.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/agents.js +0 -1256
- package/dist/agent-generator/templates/core/agents.js.map +0 -1
- package/dist/agent-generator/templates/core/architecture-rules.d.ts +0 -7
- package/dist/agent-generator/templates/core/architecture-rules.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/architecture-rules.js +0 -274
- package/dist/agent-generator/templates/core/architecture-rules.js.map +0 -1
- package/dist/agent-generator/templates/core/general-rules.d.ts +0 -8
- package/dist/agent-generator/templates/core/general-rules.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/general-rules.js +0 -301
- package/dist/agent-generator/templates/core/general-rules.js.map +0 -1
- package/dist/agent-generator/templates/core/hooks-generator.d.ts +0 -21
- package/dist/agent-generator/templates/core/hooks-generator.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/hooks-generator.js +0 -233
- package/dist/agent-generator/templates/core/hooks-generator.js.map +0 -1
- package/dist/agent-generator/templates/core/index-md.d.ts +0 -7
- package/dist/agent-generator/templates/core/index-md.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/index-md.js +0 -246
- package/dist/agent-generator/templates/core/index-md.js.map +0 -1
- package/dist/agent-generator/templates/core/orchestrator.d.ts +0 -8
- package/dist/agent-generator/templates/core/orchestrator.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/orchestrator.js +0 -422
- package/dist/agent-generator/templates/core/orchestrator.js.map +0 -1
- package/dist/agent-generator/templates/core/preflight.d.ts +0 -8
- package/dist/agent-generator/templates/core/preflight.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/preflight.js +0 -213
- package/dist/agent-generator/templates/core/preflight.js.map +0 -1
- package/dist/agent-generator/templates/core/quality-gates.d.ts +0 -11
- package/dist/agent-generator/templates/core/quality-gates.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/quality-gates.js +0 -254
- package/dist/agent-generator/templates/core/quality-gates.js.map +0 -1
- package/dist/agent-generator/templates/core/security-rules.d.ts +0 -7
- package/dist/agent-generator/templates/core/security-rules.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/security-rules.js +0 -528
- package/dist/agent-generator/templates/core/security-rules.js.map +0 -1
- package/dist/agent-generator/templates/core/skills-generator.d.ts +0 -19
- package/dist/agent-generator/templates/core/skills-generator.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/skills-generator.js +0 -546
- package/dist/agent-generator/templates/core/skills-generator.js.map +0 -1
- package/dist/agent-generator/templates/core/workflow-fix-bug.d.ts +0 -7
- package/dist/agent-generator/templates/core/workflow-fix-bug.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/workflow-fix-bug.js +0 -237
- package/dist/agent-generator/templates/core/workflow-fix-bug.js.map +0 -1
- package/dist/agent-generator/templates/core/workflow-new-feature.d.ts +0 -8
- package/dist/agent-generator/templates/core/workflow-new-feature.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/workflow-new-feature.js +0 -321
- package/dist/agent-generator/templates/core/workflow-new-feature.js.map +0 -1
- package/dist/agent-generator/templates/core/workflow-review.d.ts +0 -7
- package/dist/agent-generator/templates/core/workflow-review.d.ts.map +0 -1
- package/dist/agent-generator/templates/core/workflow-review.js +0 -104
- package/dist/agent-generator/templates/core/workflow-review.js.map +0 -1
- package/dist/agent-generator/templates/domain/index.d.ts +0 -22
- package/dist/agent-generator/templates/domain/index.d.ts.map +0 -1
- package/dist/agent-generator/templates/domain/index.js +0 -1176
- package/dist/agent-generator/templates/domain/index.js.map +0 -1
- package/dist/agent-generator/templates/stack/index.d.ts +0 -8
- package/dist/agent-generator/templates/stack/index.d.ts.map +0 -1
- package/dist/agent-generator/templates/stack/index.js +0 -695
- package/dist/agent-generator/templates/stack/index.js.map +0 -1
- package/dist/agent-generator/templates/template-helpers.d.ts +0 -75
- package/dist/agent-generator/templates/template-helpers.d.ts.map +0 -1
- package/dist/agent-generator/templates/template-helpers.js +0 -726
- package/dist/agent-generator/templates/template-helpers.js.map +0 -1
- package/dist/agent-generator/types.d.ts +0 -196
- package/dist/agent-generator/types.d.ts.map +0 -1
- package/dist/agent-generator/types.js +0 -27
- package/dist/agent-generator/types.js.map +0 -1
- package/dist/analyzer.d.ts +0 -38
- package/dist/analyzer.d.ts.map +0 -1
- package/dist/analyzer.js +0 -383
- package/dist/analyzer.js.map +0 -1
- package/dist/analyzers/forecast.d.ts +0 -85
- package/dist/analyzers/forecast.d.ts.map +0 -1
- package/dist/analyzers/forecast.js +0 -337
- package/dist/analyzers/forecast.js.map +0 -1
- package/dist/analyzers/git-cache.d.ts +0 -7
- package/dist/analyzers/git-cache.d.ts.map +0 -1
- package/dist/analyzers/git-cache.js +0 -41
- package/dist/analyzers/git-cache.js.map +0 -1
- package/dist/analyzers/git-history.d.ts +0 -113
- package/dist/analyzers/git-history.d.ts.map +0 -1
- package/dist/analyzers/git-history.js +0 -333
- package/dist/analyzers/git-history.js.map +0 -1
- package/dist/analyzers/index.d.ts +0 -10
- package/dist/analyzers/index.d.ts.map +0 -1
- package/dist/analyzers/index.js +0 -7
- package/dist/analyzers/index.js.map +0 -1
- package/dist/analyzers/temporal-scorer.d.ts +0 -72
- package/dist/analyzers/temporal-scorer.d.ts.map +0 -1
- package/dist/analyzers/temporal-scorer.js +0 -140
- package/dist/analyzers/temporal-scorer.js.map +0 -1
- package/dist/anti-patterns.d.ts +0 -24
- package/dist/anti-patterns.d.ts.map +0 -1
- package/dist/anti-patterns.js +0 -230
- package/dist/anti-patterns.js.map +0 -1
- package/dist/cli.d.ts.map +0 -1
- package/dist/cli.js.map +0 -1
- package/dist/config.d.ts +0 -12
- package/dist/config.d.ts.map +0 -1
- package/dist/config.js +0 -110
- package/dist/config.js.map +0 -1
- package/dist/diagram.d.ts +0 -9
- package/dist/diagram.d.ts.map +0 -1
- package/dist/diagram.js +0 -116
- package/dist/diagram.js.map +0 -1
- package/dist/html-reporter.d.ts +0 -47
- package/dist/html-reporter.d.ts.map +0 -1
- package/dist/html-reporter.js +0 -1747
- package/dist/html-reporter.js.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/project-summarizer.d.ts +0 -38
- package/dist/project-summarizer.d.ts.map +0 -1
- package/dist/project-summarizer.js +0 -463
- package/dist/project-summarizer.js.map +0 -1
- package/dist/refactor-engine.d.ts +0 -18
- package/dist/refactor-engine.d.ts.map +0 -1
- package/dist/refactor-engine.js +0 -86
- package/dist/refactor-engine.js.map +0 -1
- package/dist/refactor-reporter.d.ts.map +0 -1
- package/dist/refactor-reporter.js.map +0 -1
- package/dist/reporter.d.ts.map +0 -1
- package/dist/reporter.js.map +0 -1
- package/dist/rules/barrel-optimizer.d.ts +0 -13
- package/dist/rules/barrel-optimizer.d.ts.map +0 -1
- package/dist/rules/barrel-optimizer.js +0 -77
- package/dist/rules/barrel-optimizer.js.map +0 -1
- package/dist/rules/dead-code-detector.d.ts +0 -21
- package/dist/rules/dead-code-detector.d.ts.map +0 -1
- package/dist/rules/dead-code-detector.js +0 -117
- package/dist/rules/dead-code-detector.js.map +0 -1
- package/dist/rules/hub-splitter.d.ts +0 -13
- package/dist/rules/hub-splitter.d.ts.map +0 -1
- package/dist/rules/hub-splitter.js +0 -110
- package/dist/rules/hub-splitter.js.map +0 -1
- package/dist/rules/import-organizer.d.ts +0 -13
- package/dist/rules/import-organizer.d.ts.map +0 -1
- package/dist/rules/import-organizer.js +0 -85
- package/dist/rules/import-organizer.js.map +0 -1
- package/dist/rules/module-grouper.d.ts +0 -13
- package/dist/rules/module-grouper.d.ts.map +0 -1
- package/dist/rules/module-grouper.js +0 -110
- package/dist/rules/module-grouper.js.map +0 -1
- package/dist/scanner.d.ts +0 -31
- package/dist/scanner.d.ts.map +0 -1
- package/dist/scanner.js +0 -328
- package/dist/scanner.js.map +0 -1
- package/dist/scorer.d.ts +0 -27
- package/dist/scorer.d.ts.map +0 -1
- package/dist/scorer.js +0 -229
- package/dist/scorer.js.map +0 -1
- package/dist/types.d.ts +0 -186
- package/dist/types.d.ts.map +0 -1
- package/dist/types.js +0 -2
- package/dist/types.js.map +0 -1
- package/examples/sample-report.md +0 -207
- package/jest.config.js +0 -18
- package/src/agent-generator/context-enricher.ts +0 -672
- package/src/agent-generator/domain-inferrer.ts +0 -635
- package/src/agent-generator/framework-detector.ts +0 -669
- package/src/agent-generator/index.ts +0 -634
- package/src/agent-generator/stack-detector.ts +0 -115
- package/src/agent-generator/templates/core/agents.ts +0 -1296
- package/src/agent-generator/templates/core/architecture-rules.ts +0 -287
- package/src/agent-generator/templates/core/general-rules.ts +0 -306
- package/src/agent-generator/templates/core/hooks-generator.ts +0 -242
- package/src/agent-generator/templates/core/index-md.ts +0 -260
- package/src/agent-generator/templates/core/orchestrator.ts +0 -459
- package/src/agent-generator/templates/core/preflight.ts +0 -215
- package/src/agent-generator/templates/core/quality-gates.ts +0 -256
- package/src/agent-generator/templates/core/security-rules.ts +0 -543
- package/src/agent-generator/templates/core/skills-generator.ts +0 -585
- package/src/agent-generator/templates/core/workflow-fix-bug.ts +0 -239
- package/src/agent-generator/templates/core/workflow-new-feature.ts +0 -323
- package/src/agent-generator/templates/core/workflow-review.ts +0 -106
- package/src/agent-generator/templates/domain/index.ts +0 -1201
- package/src/agent-generator/templates/stack/index.ts +0 -705
- package/src/agent-generator/templates/template-helpers.ts +0 -776
- package/src/agent-generator/types.ts +0 -232
- package/src/analyzer.ts +0 -447
- package/src/analyzers/forecast.ts +0 -496
- package/src/analyzers/git-cache.ts +0 -52
- package/src/analyzers/git-history.ts +0 -488
- package/src/analyzers/index.ts +0 -33
- package/src/analyzers/temporal-scorer.ts +0 -227
- package/src/anti-patterns.ts +0 -287
- package/src/cli.ts +0 -517
- package/src/config.ts +0 -123
- package/src/diagram.ts +0 -144
- package/src/html-reporter.ts +0 -1830
- package/src/project-summarizer.ts +0 -521
- package/src/refactor-engine.ts +0 -117
- package/src/rules/barrel-optimizer.ts +0 -97
- package/src/rules/dead-code-detector.ts +0 -132
- package/src/rules/hub-splitter.ts +0 -123
- package/src/rules/import-organizer.ts +0 -98
- package/src/rules/module-grouper.ts +0 -124
- package/src/scanner.ts +0 -344
- package/src/scorer.ts +0 -254
- package/src/types.ts +0 -193
- package/tests/agent-generator.test.ts +0 -427
- package/tests/analyzers-integration.test.ts +0 -174
- package/tests/anti-patterns.test.ts +0 -94
- package/tests/context-enricher.test.ts +0 -971
- package/tests/fixtures/monorepo/package.json +0 -6
- package/tests/fixtures/monorepo/packages/app/package.json +0 -12
- package/tests/fixtures/monorepo/packages/app/src/index.ts +0 -6
- package/tests/fixtures/monorepo/packages/core/package.json +0 -7
- package/tests/fixtures/monorepo/packages/core/src/index.ts +0 -7
- package/tests/forecast.test.ts +0 -509
- package/tests/framework-detector.test.ts +0 -1172
- package/tests/git-history.test.ts +0 -254
- package/tests/monorepo-scan.test.ts +0 -170
- package/tests/scanner.test.ts +0 -54
- package/tests/scorer.test.ts +0 -674
- package/tests/stack-detector.test.ts +0 -241
- package/tests/template-generation.test.ts +0 -706
- package/tests/template-helpers.test.ts +0 -1152
- package/tests/temporal-scorer.test.ts +0 -307
- /package/dist/{reporter.js → src/adapters/reporter.js} +0 -0
|
@@ -0,0 +1,492 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Architect CLI v8.1
|
|
5
|
+
* Enterprise Architecture Analysis — @girardelli/architect
|
|
6
|
+
*
|
|
7
|
+
* Uso:
|
|
8
|
+
* npx architect analyze ./src
|
|
9
|
+
* npx architect analyze ./src --format html --output report.html
|
|
10
|
+
* npx architect diagram ./src
|
|
11
|
+
* npx architect score ./src
|
|
12
|
+
* npx architect anti-patterns ./src
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
import { architect, ProgressEvent } from '../core/architect.js';
|
|
16
|
+
import { GenesisTerminal } from '../core/GenesisTerminal.js';
|
|
17
|
+
import { c, ProgressReporter } from './progress-logger.js';
|
|
18
|
+
import { AgentExecutor } from '@girardelli/architect-agents/src/core/agent-runtime/executor.js';
|
|
19
|
+
import { ReportGenerator } from './reporter.js';
|
|
20
|
+
import { HtmlReportGenerator } from './html-reporter.js';
|
|
21
|
+
import { RefactorReportGenerator } from './refactor-reporter.js';
|
|
22
|
+
import { writeFileSync, existsSync, readFileSync } from 'fs';
|
|
23
|
+
import { resolve, basename, join } from 'path';
|
|
24
|
+
import { logger } from '@girardelli/architect-core/src/infrastructure/logger.js';
|
|
25
|
+
import { i18n } from '@girardelli/architect-core/src/core/i18n.js';
|
|
26
|
+
import * as yaml from 'yaml';
|
|
27
|
+
import chokidar from 'chokidar';
|
|
28
|
+
import { ArchitectRules, ValidationResult } from '@girardelli/architect-core/src/core/types/architect-rules.js';
|
|
29
|
+
import { RulesEngine } from '@girardelli/architect-core/src/core/rules-engine.js';
|
|
30
|
+
import { execSync } from 'child_process';
|
|
31
|
+
import { GithubActionAdapter } from './github-action.js';
|
|
32
|
+
import * as github from '@actions/github';
|
|
33
|
+
|
|
34
|
+
type OutputFormat = 'json' | 'markdown' | 'html';
|
|
35
|
+
|
|
36
|
+
interface CliOptions {
|
|
37
|
+
command: string;
|
|
38
|
+
path: string;
|
|
39
|
+
format: OutputFormat;
|
|
40
|
+
output?: string;
|
|
41
|
+
verbose: boolean;
|
|
42
|
+
locale: string;
|
|
43
|
+
watch: boolean;
|
|
44
|
+
auto: boolean;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
|
|
48
|
+
|
|
49
|
+
// ── CLI Parsing ──
|
|
50
|
+
|
|
51
|
+
|
|
52
|
+
|
|
53
|
+
function parseArgs(args: string[]): CliOptions {
|
|
54
|
+
const command = args[0] || 'analyze';
|
|
55
|
+
const pathArg = args.find((a) => !a.startsWith('--') && a !== command) || '.';
|
|
56
|
+
const formatIdx = args.indexOf('--format');
|
|
57
|
+
const format = (formatIdx >= 0 ? args[formatIdx + 1] : 'html') as OutputFormat;
|
|
58
|
+
const outputIdx = args.indexOf('--output');
|
|
59
|
+
const output = outputIdx >= 0 ? args[outputIdx + 1] : undefined;
|
|
60
|
+
const verbose = args.includes('--verbose') || args.includes('-v');
|
|
61
|
+
|
|
62
|
+
const localeIdx = args.indexOf('--locale');
|
|
63
|
+
let locale = 'en'; // default
|
|
64
|
+
if (localeIdx >= 0 && args[localeIdx + 1]) {
|
|
65
|
+
locale = args[localeIdx + 1];
|
|
66
|
+
} else {
|
|
67
|
+
// Attempt detect via ENV
|
|
68
|
+
const envLang = process.env.LANG || process.env.LANGUAGE || '';
|
|
69
|
+
if (envLang.toLowerCase().includes('pt')) {
|
|
70
|
+
locale = 'pt-BR';
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
const watch = args.includes('--watch') || args.includes('-w');
|
|
75
|
+
const auto = args.includes('--auto');
|
|
76
|
+
|
|
77
|
+
return { command, path: resolve(pathArg), format, output, verbose, locale, watch, auto };
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
function printUsage(): void {
|
|
81
|
+
console.log(`
|
|
82
|
+
${c.cyan}${c.bold}⚡ Architect Agent v7.0${c.reset} — Enterprise Architecture Intelligence
|
|
83
|
+
|
|
84
|
+
${c.bold}Usage:${c.reset}
|
|
85
|
+
architect <command> [path] [options]
|
|
86
|
+
|
|
87
|
+
${c.bold}Commands:${c.reset}
|
|
88
|
+
${c.cyan}analyze${c.reset} Full architecture analysis (default)
|
|
89
|
+
${c.cyan}genesis${c.reset} [NEW] Interactive God Mode Terminal (TUI)
|
|
90
|
+
${c.cyan}execute${c.reset} Autonomous Agent Runtime - auto-refactor project
|
|
91
|
+
${c.cyan}check${c.reset} Run architecture-as-code validation against .architect.rules.yml
|
|
92
|
+
${c.cyan}pr-review${c.reset} Run in GitHub Actions to comment on PRs with Score Delta
|
|
93
|
+
${c.cyan}refactor${c.reset} Generate refactoring plan with actionable steps
|
|
94
|
+
${c.cyan}agents${c.reset} Generate/audit .agent/ directory with AI agents
|
|
95
|
+
${c.cyan}diagram${c.reset} Generate architecture diagram only
|
|
96
|
+
${c.cyan}score${c.reset} Calculate quality score only
|
|
97
|
+
${c.cyan}anti-patterns${c.reset} Detect anti-patterns only
|
|
98
|
+
${c.cyan}layers${c.reset} Analyze layer structure only
|
|
99
|
+
|
|
100
|
+
${c.bold}Options:${c.reset}
|
|
101
|
+
--format <type> Output format: html, json, markdown (default: html)
|
|
102
|
+
--output <file> Output file path
|
|
103
|
+
--locale <lang> Language (en or pt-BR)
|
|
104
|
+
--verbose, -v Enable verbose debug logging
|
|
105
|
+
--watch, -w Watch mode (re-run check on file changes)
|
|
106
|
+
--auto Agent YOLO Mode - auto-approve structural changes
|
|
107
|
+
--help Show this help message
|
|
108
|
+
|
|
109
|
+
${c.bold}Examples:${c.reset}
|
|
110
|
+
${c.dim}$${c.reset} architect execute ./src
|
|
111
|
+
${c.dim}$${c.reset} architect execute ./src --auto
|
|
112
|
+
${c.dim}$${c.reset} architect analyze ./src --format html --output report.html
|
|
113
|
+
|
|
114
|
+
${c.dim}@girardelli/architect — Girardelli Tecnologia${c.reset}
|
|
115
|
+
`);
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
async function main(): Promise<void> {
|
|
119
|
+
const args = process.argv.slice(2);
|
|
120
|
+
|
|
121
|
+
if (args.includes('--help') || args.includes('-h') || args.length === 0) {
|
|
122
|
+
printUsage();
|
|
123
|
+
process.exit(0);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
const options = parseArgs(args);
|
|
127
|
+
logger.setup({ verbose: options.verbose, json: options.format === 'json' });
|
|
128
|
+
|
|
129
|
+
// Set global locale
|
|
130
|
+
i18n.setLocale(options.locale as any);
|
|
131
|
+
|
|
132
|
+
try {
|
|
133
|
+
switch (options.command) {
|
|
134
|
+
case 'genesis': {
|
|
135
|
+
const terminal = new GenesisTerminal(architect);
|
|
136
|
+
await terminal.start();
|
|
137
|
+
break;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
case 'analyze': {
|
|
141
|
+
const progress = new ProgressReporter();
|
|
142
|
+
progress.printHeader(options.path);
|
|
143
|
+
|
|
144
|
+
const report = await architect.analyze(options.path, (e: ProgressEvent) => progress.onProgress(e));
|
|
145
|
+
|
|
146
|
+
// Refactoring
|
|
147
|
+
progress.printExtraPhase('REFACTOR ENGINE', 'Building refactoring plan', c.orange);
|
|
148
|
+
const plan = architect.refactor(report, options.path);
|
|
149
|
+
progress.printExtraComplete(
|
|
150
|
+
`${c.white}${plan.steps.length}${c.reset}${c.dim} steps · ${c.reset}${c.white}${plan.totalOperations}${c.reset}${c.dim} operations · est. +${plan.estimatedScoreAfter.overall - plan.currentScore.overall} pts${c.reset}`
|
|
151
|
+
);
|
|
152
|
+
|
|
153
|
+
// Agent suggestion
|
|
154
|
+
progress.printExtraPhase('AGENT SYSTEM', 'Analyzing agent requirements', c.magenta);
|
|
155
|
+
const agentSuggestion = architect.suggestAgents(report, plan, options.path);
|
|
156
|
+
progress.printExtraComplete(
|
|
157
|
+
`${c.white}${agentSuggestion.suggestedAgents.length}${c.reset}${c.dim} agents suggested${agentSuggestion.hasExistingAgents ? ' (existing .agent/ audited)' : ''}${c.reset}`
|
|
158
|
+
);
|
|
159
|
+
|
|
160
|
+
const projectName = String(report.projectInfo.name || basename(options.path)).replace(/[^a-zA-Z0-9-]/g, '-');
|
|
161
|
+
|
|
162
|
+
// Report generation
|
|
163
|
+
if (options.format === 'html') {
|
|
164
|
+
progress.printExtraPhase('REPORT BUILDER', 'Generating interactive HTML report', c.cyan);
|
|
165
|
+
const htmlGenerator = new HtmlReportGenerator();
|
|
166
|
+
const html = htmlGenerator.generateHtml(report, plan, agentSuggestion);
|
|
167
|
+
const outputPath = options.output || `architect-report-${projectName}.html`;
|
|
168
|
+
writeFileSync(outputPath, html);
|
|
169
|
+
progress.printExtraComplete(`${c.green}${outputPath}${c.reset}`);
|
|
170
|
+
} else if (options.format === 'markdown') {
|
|
171
|
+
progress.printExtraPhase('REPORT BUILDER', 'Generating Markdown report', c.cyan);
|
|
172
|
+
const mdGenerator = new ReportGenerator();
|
|
173
|
+
const markdown = mdGenerator.generateMarkdownReport(report);
|
|
174
|
+
const outputPath = options.output || `architect-report-${projectName}.md`;
|
|
175
|
+
writeFileSync(outputPath, markdown);
|
|
176
|
+
progress.printExtraComplete(`${c.green}${outputPath}${c.reset}`);
|
|
177
|
+
} else {
|
|
178
|
+
progress.printExtraPhase('REPORT BUILDER', 'Generating JSON report', c.cyan);
|
|
179
|
+
const outputPath = options.output || `architect-report-${projectName}.json`;
|
|
180
|
+
writeFileSync(outputPath, JSON.stringify({ report, plan, agentSuggestion }, null, 2));
|
|
181
|
+
progress.printExtraComplete(`${c.green}${outputPath}${c.reset}`);
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
// Summary
|
|
185
|
+
progress.printSummary(report.score.overall, report.score.breakdown, {
|
|
186
|
+
files: report.projectInfo.totalFiles,
|
|
187
|
+
lines: report.projectInfo.totalLines,
|
|
188
|
+
antiPatterns: report.antiPatterns.length,
|
|
189
|
+
refactorSteps: plan.steps.length,
|
|
190
|
+
refactorOps: plan.totalOperations,
|
|
191
|
+
agents: agentSuggestion.suggestedAgents.length,
|
|
192
|
+
});
|
|
193
|
+
break;
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
case 'refactor': {
|
|
197
|
+
const progress = new ProgressReporter();
|
|
198
|
+
progress.printHeader(options.path);
|
|
199
|
+
|
|
200
|
+
const report = await architect.analyze(options.path, (e: ProgressEvent) => progress.onProgress(e));
|
|
201
|
+
|
|
202
|
+
progress.printExtraPhase('REFACTOR ENGINE', 'Building refactoring plan', c.orange);
|
|
203
|
+
const plan = architect.refactor(report, options.path);
|
|
204
|
+
const projectName = String(report.projectInfo.name || basename(options.path)).replace(/[^a-zA-Z0-9-]/g, '-');
|
|
205
|
+
|
|
206
|
+
if (options.format === 'json') {
|
|
207
|
+
const outputPath = options.output || `refactor-plan-${projectName}.json`;
|
|
208
|
+
writeFileSync(outputPath, JSON.stringify(plan, null, 2));
|
|
209
|
+
progress.printExtraComplete(`${c.green}${outputPath}${c.reset}`);
|
|
210
|
+
} else {
|
|
211
|
+
const refactorReporter = new RefactorReportGenerator();
|
|
212
|
+
const html = refactorReporter.generateHtml(plan);
|
|
213
|
+
const outputPath = options.output || `refactor-plan-${projectName}.html`;
|
|
214
|
+
writeFileSync(outputPath, html);
|
|
215
|
+
progress.printExtraComplete(`${c.green}${outputPath}${c.reset}`);
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
process.stderr.write(`\n ${c.bold}REFACTORING PLAN${c.reset}\n`);
|
|
219
|
+
process.stderr.write(` ${c.dim}Steps:${c.reset} ${c.white}${plan.steps.length}${c.reset} ${c.dim}Ops:${c.reset} ${c.white}${plan.totalOperations}${c.reset} ${c.dim}Tier1:${c.reset} ${c.white}${plan.tier1Steps}${c.reset} ${c.dim}Tier2:${c.reset} ${c.white}${plan.tier2Steps}${c.reset}\n`);
|
|
220
|
+
process.stderr.write(` ${c.dim}Score:${c.reset} ${c.white}${plan.currentScore.overall}${c.reset}${c.dim} → ${c.reset}${c.green}${plan.estimatedScoreAfter.overall}${c.reset} ${c.dim}(+${plan.estimatedScoreAfter.overall - plan.currentScore.overall})${c.reset}\n\n`);
|
|
221
|
+
break;
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
case 'check': {
|
|
225
|
+
const rulesPath = join(options.path, '.architect.rules.yml');
|
|
226
|
+
|
|
227
|
+
const executeCheck = async (): Promise<boolean> => {
|
|
228
|
+
if (!existsSync(rulesPath)) {
|
|
229
|
+
logger.error(`Rules file not found at: ${rulesPath}`);
|
|
230
|
+
logger.error(`Create a '.architect.rules.yml' file to use the 'check' command.`);
|
|
231
|
+
return false;
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
let rules: ArchitectRules;
|
|
235
|
+
try {
|
|
236
|
+
const raw = readFileSync(rulesPath, 'utf8');
|
|
237
|
+
rules = yaml.parse(raw) as ArchitectRules;
|
|
238
|
+
} catch (e: any) {
|
|
239
|
+
logger.error(`Failed to parse ${rulesPath}: ${e.message}`);
|
|
240
|
+
return false;
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
process.stderr.write(`\n ${c.cyan}◉${c.reset} ${c.bold}RULES ENGINE${c.reset} ${c.dim}— Validating against .architect.rules.yml...${c.reset}\n`);
|
|
244
|
+
|
|
245
|
+
const report = await architect.analyze(options.path);
|
|
246
|
+
const engine = new RulesEngine();
|
|
247
|
+
const result = engine.validate(report, rules);
|
|
248
|
+
|
|
249
|
+
if (result.violations.length === 0) {
|
|
250
|
+
process.stderr.write(`\n ${c.green}✓ All quality gates and boundaries passed!${c.reset} ${c.dim}(Score: ${report.score.overall}/100)${c.reset}\n\n`);
|
|
251
|
+
return true;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
process.stderr.write(`\n ${c.red}✗ Architecture validation failed!${c.reset} ${c.dim}(Score: ${report.score.overall}/100)${c.reset}\n\n`);
|
|
255
|
+
|
|
256
|
+
let errors = 0;
|
|
257
|
+
let warnings = 0;
|
|
258
|
+
|
|
259
|
+
for (const v of result.violations) {
|
|
260
|
+
if (v.level === 'error') {
|
|
261
|
+
errors++;
|
|
262
|
+
process.stderr.write(` ${c.red}[ERROR]${c.reset} ${c.bold}${v.rule}${c.reset}: ${v.message}\n`);
|
|
263
|
+
} else {
|
|
264
|
+
warnings++;
|
|
265
|
+
process.stderr.write(` ${c.yellow}[WARN]${c.reset} ${c.bold}${v.rule}${c.reset}: ${v.message}\n`);
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
process.stderr.write(`\n ${c.dim}Total: ${errors} errors, ${warnings} warnings.${c.reset}\n\n`);
|
|
270
|
+
return result.success;
|
|
271
|
+
};
|
|
272
|
+
|
|
273
|
+
if (options.watch) {
|
|
274
|
+
process.stderr.write(`\n ${c.dim}👀 Watch mode enabled. Listening for changes in ${options.path}...${c.reset}\n`);
|
|
275
|
+
|
|
276
|
+
await executeCheck();
|
|
277
|
+
|
|
278
|
+
const watcher = chokidar.watch([
|
|
279
|
+
join(options.path, '**/*.ts'),
|
|
280
|
+
join(options.path, '**/*.js'),
|
|
281
|
+
rulesPath
|
|
282
|
+
], {
|
|
283
|
+
ignored: /(^|[\/\\])\../, // ignore dotfiles
|
|
284
|
+
persistent: true,
|
|
285
|
+
ignoreInitial: true,
|
|
286
|
+
});
|
|
287
|
+
|
|
288
|
+
watcher.on('all', async (_event, path) => {
|
|
289
|
+
console.clear();
|
|
290
|
+
process.stderr.write(` ${c.dim}↻ File changed: ${basename(path)}. Re-running checks...${c.reset}\n`);
|
|
291
|
+
await executeCheck();
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
// Keep process alive
|
|
295
|
+
return new Promise(() => {});
|
|
296
|
+
} else {
|
|
297
|
+
const success = await executeCheck();
|
|
298
|
+
if (!success) process.exit(1);
|
|
299
|
+
}
|
|
300
|
+
break;
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
case 'pr-review': {
|
|
304
|
+
const githubToken = process.env.GITHUB_TOKEN;
|
|
305
|
+
if (!githubToken) {
|
|
306
|
+
logger.error('GITHUB_TOKEN environment variable is required for pr-review');
|
|
307
|
+
process.exit(1);
|
|
308
|
+
return;
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
if (!github.context.payload.pull_request) {
|
|
312
|
+
logger.error('Not running in a Pull Request context. Exiting.');
|
|
313
|
+
process.exit(0);
|
|
314
|
+
return;
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
process.stderr.write(`\n ${c.cyan}◉${c.reset} ${c.bold}PR REVIEWER${c.reset} ${c.dim}— Analyzing HEAD vs BASE...${c.reset}\n`);
|
|
318
|
+
|
|
319
|
+
const headReport = await architect.analyze(options.path);
|
|
320
|
+
|
|
321
|
+
let baseReport = null;
|
|
322
|
+
try {
|
|
323
|
+
const baseRef = github.context.payload.pull_request.base.ref;
|
|
324
|
+
process.stderr.write(` ${c.dim}↳ Fetching and checking out base branch: ${baseRef}${c.reset}\n`);
|
|
325
|
+
|
|
326
|
+
execSync(`git fetch origin ${baseRef} || true`, { stdio: 'ignore', cwd: options.path });
|
|
327
|
+
const currentBranch = execSync(`git rev-parse --abbrev-ref HEAD`, { cwd: options.path }).toString().trim();
|
|
328
|
+
execSync(`git checkout ${baseRef}`, { stdio: 'ignore', cwd: options.path });
|
|
329
|
+
|
|
330
|
+
process.stderr.write(` ${c.dim}↳ Analyzing base branch...${c.reset}\n`);
|
|
331
|
+
baseReport = await architect.analyze(options.path);
|
|
332
|
+
|
|
333
|
+
process.stderr.write(` ${c.dim}↳ Reverting to original branch: ${currentBranch}${c.reset}\n`);
|
|
334
|
+
execSync(`git checkout ${currentBranch}`, { stdio: 'ignore', cwd: options.path });
|
|
335
|
+
} catch (e: any) {
|
|
336
|
+
logger.error(`Failed to analyze base branch: ${e.message}`);
|
|
337
|
+
process.stderr.write(` ${c.yellow}⚠️ Falling back to single-scan (no delta)${c.reset}\n`);
|
|
338
|
+
// Ensure we try to checkout back just in case
|
|
339
|
+
try { execSync(`git checkout -`, { stdio: 'ignore', cwd: options.path }); } catch {}
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
let validationResult: ValidationResult | undefined;
|
|
343
|
+
const rulesPath = join(options.path, '.architect.rules.yml');
|
|
344
|
+
if (existsSync(rulesPath)) {
|
|
345
|
+
process.stderr.write(` ${c.dim}↳ Validating Architecture Rules...${c.reset}\n`);
|
|
346
|
+
const raw = readFileSync(rulesPath, 'utf8');
|
|
347
|
+
const rules = yaml.parse(raw) as ArchitectRules;
|
|
348
|
+
const engine = new RulesEngine();
|
|
349
|
+
validationResult = engine.validate(headReport, rules);
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
process.stderr.write(` ${c.dim}↳ Posting comment to GitHub PR #${github.context.payload.pull_request.number}...${c.reset}\n`);
|
|
353
|
+
const adapter = new GithubActionAdapter(githubToken);
|
|
354
|
+
await adapter.postComment(headReport, baseReport, validationResult);
|
|
355
|
+
|
|
356
|
+
process.stderr.write(`\n ${c.green}✓ PR Comment successfully posted!${c.reset}\n\n`);
|
|
357
|
+
|
|
358
|
+
if (validationResult && !validationResult.success) {
|
|
359
|
+
process.stderr.write(` ${c.red}✗ Architecture rules validation failed (Action blocked).${c.reset}\n\n`);
|
|
360
|
+
process.exit(1);
|
|
361
|
+
}
|
|
362
|
+
break;
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
case 'agents': {
|
|
366
|
+
const progress = new ProgressReporter();
|
|
367
|
+
progress.printHeader(options.path);
|
|
368
|
+
|
|
369
|
+
const report = await architect.analyze(options.path, (e: ProgressEvent) => progress.onProgress(e));
|
|
370
|
+
|
|
371
|
+
progress.printExtraPhase('REFACTOR ENGINE', 'Building refactoring plan', c.orange);
|
|
372
|
+
const plan = architect.refactor(report, options.path);
|
|
373
|
+
progress.printExtraComplete(`${c.white}${plan.steps.length}${c.reset}${c.dim} steps${c.reset}`);
|
|
374
|
+
|
|
375
|
+
progress.printExtraPhase('AGENT GENERATOR', 'Generating .agent/ framework', c.magenta);
|
|
376
|
+
const outputDir = options.output || undefined;
|
|
377
|
+
const result = architect.agents(report, plan, options.path, outputDir);
|
|
378
|
+
progress.printExtraComplete(`${c.white}${result.generated.length}${c.reset}${c.dim} files generated${c.reset}`);
|
|
379
|
+
|
|
380
|
+
process.stderr.write(`\n ${c.bold}🤖 AGENT SYSTEM${c.reset}\n\n`);
|
|
381
|
+
|
|
382
|
+
if (result.generated.length > 0) {
|
|
383
|
+
process.stderr.write(` ${c.green}Generated:${c.reset}\n`);
|
|
384
|
+
for (const file of result.generated) {
|
|
385
|
+
process.stderr.write(` ${c.dim} 📄${c.reset} ${file}\n`);
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
if (result.audit.length > 0) {
|
|
390
|
+
const missing = result.audit.filter((f: any) => f.type === 'MISSING');
|
|
391
|
+
const improvements = result.audit.filter((f: any) => f.type === 'IMPROVEMENT');
|
|
392
|
+
const ok = result.audit.filter((f: any) => f.type === 'OK');
|
|
393
|
+
|
|
394
|
+
if (ok.length > 0) process.stderr.write(`\n ${c.green}✓ ${ok.length} checks passed${c.reset}\n`);
|
|
395
|
+
if (missing.length > 0) {
|
|
396
|
+
process.stderr.write(`\n ${c.red}✗ ${missing.length} missing (auto-generated):${c.reset}\n`);
|
|
397
|
+
for (const f of missing) process.stderr.write(` ${c.dim}📄${c.reset} ${f.file} — ${f.description}\n`);
|
|
398
|
+
}
|
|
399
|
+
if (improvements.length > 0) {
|
|
400
|
+
process.stderr.write(`\n ${c.yellow}💡 ${improvements.length} improvements:${c.reset}\n`);
|
|
401
|
+
for (const f of improvements) {
|
|
402
|
+
process.stderr.write(` ${c.dim}⚡${c.reset} ${f.description}\n`);
|
|
403
|
+
if (f.suggestion) process.stderr.write(` ${c.dim}→ ${f.suggestion}${c.reset}\n`);
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
}
|
|
407
|
+
process.stderr.write(`\n ${c.dim}Score: ${report.score.overall}/100${c.reset}\n\n`);
|
|
408
|
+
break;
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
case 'execute': {
|
|
412
|
+
const progress = new ProgressReporter();
|
|
413
|
+
progress.printHeader(options.path);
|
|
414
|
+
|
|
415
|
+
const report = await architect.analyze(options.path, (e: ProgressEvent) => progress.onProgress(e));
|
|
416
|
+
|
|
417
|
+
progress.printExtraPhase('REFACTOR ENGINE', 'Building automated refactoring plan', c.orange);
|
|
418
|
+
const plan = architect.refactor(report, options.path);
|
|
419
|
+
progress.printExtraComplete(`${c.white}${plan.steps.length}${c.reset}${c.dim} steps generated${c.reset}`);
|
|
420
|
+
|
|
421
|
+
const executor = new AgentExecutor(options.auto);
|
|
422
|
+
await executor.executePlan(plan);
|
|
423
|
+
break;
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
case 'diagram': {
|
|
427
|
+
const diagram = await architect.diagram(options.path);
|
|
428
|
+
if (options.output) {
|
|
429
|
+
writeFileSync(options.output, diagram);
|
|
430
|
+
process.stderr.write(` ${c.green}✓${c.reset} Diagram saved: ${options.output}\n`);
|
|
431
|
+
} else {
|
|
432
|
+
console.log(diagram);
|
|
433
|
+
}
|
|
434
|
+
break;
|
|
435
|
+
}
|
|
436
|
+
|
|
437
|
+
case 'score': {
|
|
438
|
+
const score = await architect.score(options.path);
|
|
439
|
+
if (options.format === 'json') {
|
|
440
|
+
console.log(JSON.stringify(score, null, 2));
|
|
441
|
+
} else {
|
|
442
|
+
const scoreColor = score.overall >= 80 ? c.green : score.overall >= 60 ? c.yellow : c.red;
|
|
443
|
+
process.stderr.write(`\n ${c.bold}ARCHITECTURE SCORE${c.reset}\n`);
|
|
444
|
+
process.stderr.write(` ${scoreColor}${c.bold}${score.overall}/100${c.reset}\n\n`);
|
|
445
|
+
for (const [name, value] of Object.entries(score.breakdown)) {
|
|
446
|
+
process.stderr.write(` ${c.dim}${name}:${c.reset} ${c.white}${value}${c.reset}\n`);
|
|
447
|
+
}
|
|
448
|
+
process.stderr.write('\n');
|
|
449
|
+
}
|
|
450
|
+
break;
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
case 'anti-patterns': {
|
|
454
|
+
const patterns = await architect.antiPatterns(options.path);
|
|
455
|
+
if (options.format === 'json') {
|
|
456
|
+
console.log(JSON.stringify(patterns, null, 2));
|
|
457
|
+
} else {
|
|
458
|
+
process.stderr.write(`\n ${c.bold}ANTI-PATTERNS${c.reset} — ${patterns.length} found\n\n`);
|
|
459
|
+
for (const p of patterns) {
|
|
460
|
+
const sevColor = p.severity === 'CRITICAL' ? c.red : p.severity === 'HIGH' ? c.orange : c.yellow;
|
|
461
|
+
process.stderr.write(` ${sevColor}[${p.severity}]${c.reset} ${c.bold}${p.name}${c.reset}: ${p.description}\n`);
|
|
462
|
+
}
|
|
463
|
+
process.stderr.write('\n');
|
|
464
|
+
}
|
|
465
|
+
break;
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
case 'layers': {
|
|
469
|
+
const layers = await architect.layers(options.path);
|
|
470
|
+
if (options.format === 'json') {
|
|
471
|
+
console.log(JSON.stringify(layers, null, 2));
|
|
472
|
+
} else {
|
|
473
|
+
process.stderr.write(`\n ${c.bold}ARCHITECTURE LAYERS${c.reset}\n\n`);
|
|
474
|
+
for (const l of layers) {
|
|
475
|
+
process.stderr.write(` ${c.cyan}${l.name}${c.reset}: ${c.white}${l.files.length}${c.reset} files\n`);
|
|
476
|
+
}
|
|
477
|
+
process.stderr.write('\n');
|
|
478
|
+
}
|
|
479
|
+
break;
|
|
480
|
+
}
|
|
481
|
+
|
|
482
|
+
default:
|
|
483
|
+
logger.error(`Unknown command: ${options.command}`);
|
|
484
|
+
process.exit(1);
|
|
485
|
+
}
|
|
486
|
+
} catch (err: any) {
|
|
487
|
+
logger.error('Fatal execution error', err);
|
|
488
|
+
process.exit(1);
|
|
489
|
+
}
|
|
490
|
+
}
|
|
491
|
+
|
|
492
|
+
main();
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
import * as core from '@actions/core';
|
|
2
|
+
import * as github from '@actions/github';
|
|
3
|
+
import { AnalysisReport } from '@girardelli/architect-core/src/core/types/core.js';
|
|
4
|
+
import { ValidationResult } from '@girardelli/architect-core/src/core/types/architect-rules.js';
|
|
5
|
+
|
|
6
|
+
export class GithubActionAdapter {
|
|
7
|
+
private octokit: ReturnType<typeof github.getOctokit>;
|
|
8
|
+
private context = github.context;
|
|
9
|
+
|
|
10
|
+
constructor(token: string) {
|
|
11
|
+
this.octokit = github.getOctokit(token);
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
async postComment(
|
|
15
|
+
headReport: AnalysisReport,
|
|
16
|
+
baseReport?: AnalysisReport | null,
|
|
17
|
+
validationResult?: ValidationResult
|
|
18
|
+
): Promise<void> {
|
|
19
|
+
if (!this.context.payload.pull_request) {
|
|
20
|
+
core.info('Not running in a Pull Request context. Skipping comment.');
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const issueNumber = this.context.payload.pull_request.number;
|
|
25
|
+
const body = this.buildMarkdownBody(headReport, baseReport, validationResult);
|
|
26
|
+
|
|
27
|
+
await this.octokit.rest.issues.createComment({
|
|
28
|
+
owner: this.context.repo.owner,
|
|
29
|
+
repo: this.context.repo.repo,
|
|
30
|
+
issue_number: issueNumber,
|
|
31
|
+
body,
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
private buildMarkdownBody(
|
|
36
|
+
headReport: AnalysisReport,
|
|
37
|
+
baseReport?: AnalysisReport | null,
|
|
38
|
+
validationResult?: ValidationResult
|
|
39
|
+
): string {
|
|
40
|
+
const hScore = headReport.score.overall;
|
|
41
|
+
const bScore = baseReport ? baseReport.score.overall : null;
|
|
42
|
+
|
|
43
|
+
let deltaStr = '';
|
|
44
|
+
if (bScore !== null) {
|
|
45
|
+
const delta = hScore - bScore;
|
|
46
|
+
if (delta > 0) {
|
|
47
|
+
deltaStr = `📈 **+${delta}** (Improved from ${bScore})`;
|
|
48
|
+
} else if (delta < 0) {
|
|
49
|
+
deltaStr = `📉 **${delta}** (Regressed from ${bScore})`;
|
|
50
|
+
} else {
|
|
51
|
+
deltaStr = `➖ **0** (Stable at ${bScore})`;
|
|
52
|
+
}
|
|
53
|
+
} else {
|
|
54
|
+
deltaStr = `🆕 (First Scan)`;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
let validationStr = '';
|
|
58
|
+
if (validationResult) {
|
|
59
|
+
if (validationResult.success) {
|
|
60
|
+
validationStr = `✅ **Quality Gates Passed!**`;
|
|
61
|
+
} else {
|
|
62
|
+
validationStr = `❌ **Quality Gates Failed!**\n\n`;
|
|
63
|
+
validationResult.violations.forEach(v => {
|
|
64
|
+
validationStr += `- **[${v.level.toUpperCase()}]** \`${v.rule}\`: ${v.message}\n`;
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Build anti-pattern warnings
|
|
70
|
+
const critical = headReport.antiPatterns.filter(a => a.severity === 'CRITICAL');
|
|
71
|
+
const high = headReport.antiPatterns.filter(a => a.severity === 'HIGH');
|
|
72
|
+
|
|
73
|
+
let antiPatternsStr = ``;
|
|
74
|
+
if (critical.length > 0 || high.length > 0) {
|
|
75
|
+
antiPatternsStr = `### ⚠️ Anti-Patterns Detected\n`;
|
|
76
|
+
critical.forEach(p => {
|
|
77
|
+
antiPatternsStr += `- 🔴 **${p.name}** in \`${p.location}\`: ${p.description}\n`;
|
|
78
|
+
});
|
|
79
|
+
high.forEach(p => {
|
|
80
|
+
antiPatternsStr += `- 🟠 **${p.name}** in \`${p.location}\`: ${p.description}\n`;
|
|
81
|
+
});
|
|
82
|
+
} else {
|
|
83
|
+
antiPatternsStr = `✅ **No Critical/High Anti-Patterns!** Code is clean.\n`;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
return `
|
|
87
|
+
## 🏗 Architect Intelligence Report
|
|
88
|
+
|
|
89
|
+
> **Architecture Score:** \`${hScore}/100\`
|
|
90
|
+
> **Delta:** ${deltaStr}
|
|
91
|
+
|
|
92
|
+
${validationStr}
|
|
93
|
+
|
|
94
|
+
---
|
|
95
|
+
### 📊 Score Breakdown
|
|
96
|
+
| Metric | Score |
|
|
97
|
+
| --- | --- |
|
|
98
|
+
| 🧩 Modularity | ${headReport.score.breakdown.modularity} |
|
|
99
|
+
| 🔗 Coupling | ${headReport.score.breakdown.coupling} |
|
|
100
|
+
| 🎯 Cohesion | ${headReport.score.breakdown.cohesion} |
|
|
101
|
+
| 🥞 Layering | ${headReport.score.breakdown.layering} |
|
|
102
|
+
|
|
103
|
+
${antiPatternsStr}
|
|
104
|
+
|
|
105
|
+
---
|
|
106
|
+
*Report generated automatically by [Architect Genesis API](https://github.com/girardelli/architect).*
|
|
107
|
+
`.trim();
|
|
108
|
+
}
|
|
109
|
+
}
|