@evolith/smart-cli 0.0.2-beta → 1.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/README.es.md +51 -15
- package/README.md +51 -15
- package/dist/app.module.js +2 -2
- package/dist/app.module.js.map +1 -1
- package/dist/app.module.spec.d.ts +1 -0
- package/dist/app.module.spec.js +305 -0
- package/dist/app.module.spec.js.map +1 -0
- package/dist/application/ports/webhook-notifier.port.d.ts +4 -0
- package/dist/application/ports/webhook-notifier.port.js +3 -0
- package/dist/application/ports/webhook-notifier.port.js.map +1 -0
- package/dist/application/services/index.d.ts +1 -0
- package/dist/application/services/index.js +28 -2
- package/dist/application/services/index.js.map +1 -1
- package/dist/application/services/initialize-project.spec.d.ts +1 -0
- package/dist/application/services/initialize-project.spec.js +211 -0
- package/dist/application/services/initialize-project.spec.js.map +1 -0
- package/dist/application/use-cases/evaluate-gate.use-case.d.ts +20 -0
- package/dist/application/use-cases/evaluate-gate.use-case.js +72 -0
- package/dist/application/use-cases/evaluate-gate.use-case.js.map +1 -0
- package/dist/application/use-cases/validate-satellite.use-case.d.ts +1 -0
- package/dist/application/use-cases/validate-satellite.use-case.js +7 -3
- package/dist/application/use-cases/validate-satellite.use-case.js.map +1 -1
- package/dist/commands/adr/adr.command.d.ts +2 -0
- package/dist/commands/adr/adr.command.js +36 -14
- package/dist/commands/adr/adr.command.js.map +1 -1
- package/dist/commands/adr/adr.command.spec.d.ts +1 -0
- package/dist/commands/adr/adr.command.spec.js +409 -0
- package/dist/commands/adr/adr.command.spec.js.map +1 -0
- package/dist/commands/architecture/scaffold.command.d.ts +1 -0
- package/dist/commands/architecture/scaffold.command.js +23 -1
- package/dist/commands/architecture/scaffold.command.js.map +1 -1
- package/dist/commands/architecture/scaffold.command.spec.d.ts +1 -0
- package/dist/commands/architecture/scaffold.command.spec.js +207 -0
- package/dist/commands/architecture/scaffold.command.spec.js.map +1 -0
- package/dist/commands/completion/completion.command.spec.d.ts +1 -0
- package/dist/commands/completion/completion.command.spec.js +240 -0
- package/dist/commands/completion/completion.command.spec.js.map +1 -0
- package/dist/commands/docs/docs.command.spec.d.ts +1 -0
- package/dist/commands/docs/docs.command.spec.js +87 -0
- package/dist/commands/docs/docs.command.spec.js.map +1 -0
- package/dist/commands/drift/drift.command.spec.d.ts +1 -0
- package/dist/commands/drift/drift.command.spec.js +327 -0
- package/dist/commands/drift/drift.command.spec.js.map +1 -0
- package/dist/commands/gate/gate.command.d.ts +23 -0
- package/dist/commands/gate/gate.command.js +210 -0
- package/dist/commands/gate/gate.command.js.map +1 -0
- package/dist/commands/history/history.command.spec.d.ts +1 -0
- package/dist/commands/history/history.command.spec.js +392 -0
- package/dist/commands/history/history.command.spec.js.map +1 -0
- package/dist/commands/init/agents.command.spec.js +398 -0
- package/dist/commands/init/agents.command.spec.js.map +1 -1
- package/dist/commands/init/init.command.spec.d.ts +1 -0
- package/dist/commands/init/init.command.spec.js +272 -0
- package/dist/commands/init/init.command.spec.js.map +1 -0
- package/dist/commands/init/upgrade.command.spec.d.ts +1 -0
- package/dist/commands/init/upgrade.command.spec.js +378 -0
- package/dist/commands/init/upgrade.command.spec.js.map +1 -0
- package/dist/commands/mcp/mcp-serve.command.spec.d.ts +1 -0
- package/dist/commands/mcp/mcp-serve.command.spec.js +58 -0
- package/dist/commands/mcp/mcp-serve.command.spec.js.map +1 -0
- package/dist/commands/sdlc/gate-status.command.d.ts +3 -1
- package/dist/commands/sdlc/gate-status.command.js +78 -3
- package/dist/commands/sdlc/gate-status.command.js.map +1 -1
- package/dist/commands/sdlc/gate-status.command.spec.d.ts +1 -0
- package/dist/commands/sdlc/gate-status.command.spec.js +302 -0
- package/dist/commands/sdlc/gate-status.command.spec.js.map +1 -0
- package/dist/commands/sdlc/generate-domain.command.d.ts +2 -0
- package/dist/commands/sdlc/generate-domain.command.js +123 -12
- package/dist/commands/sdlc/generate-domain.command.js.map +1 -1
- package/dist/commands/sdlc/generate-domain.command.spec.d.ts +1 -0
- package/dist/commands/sdlc/generate-domain.command.spec.js +121 -0
- package/dist/commands/sdlc/generate-domain.command.spec.js.map +1 -0
- package/dist/commands/sdlc/handoff.command.js +2 -2
- package/dist/commands/sdlc/handoff.command.js.map +1 -1
- package/dist/commands/sdlc/handoff.command.spec.d.ts +1 -0
- package/dist/commands/sdlc/handoff.command.spec.js +395 -0
- package/dist/commands/sdlc/handoff.command.spec.js.map +1 -0
- package/dist/commands/sdlc/sdlc.command.js +40 -6
- package/dist/commands/sdlc/sdlc.command.js.map +1 -1
- package/dist/commands/sdlc/sdlc.command.spec.d.ts +1 -0
- package/dist/commands/sdlc/sdlc.command.spec.js +79 -0
- package/dist/commands/sdlc/sdlc.command.spec.js.map +1 -0
- package/dist/commands/standards/standards.command.spec.d.ts +1 -0
- package/dist/commands/standards/standards.command.spec.js +311 -0
- package/dist/commands/standards/standards.command.spec.js.map +1 -0
- package/dist/commands/validate/validate.command.d.ts +2 -0
- package/dist/commands/validate/validate.command.js +16 -2
- package/dist/commands/validate/validate.command.js.map +1 -1
- package/dist/commands/validate/validate.command.spec.d.ts +1 -0
- package/dist/commands/validate/validate.command.spec.js +368 -0
- package/dist/commands/validate/validate.command.spec.js.map +1 -0
- package/dist/core/abstractions/interfaces.d.ts +1 -0
- package/dist/core/abstractions/interfaces.js.map +1 -1
- package/dist/core/abstractions/providers/config-parser.provider.spec.d.ts +1 -0
- package/dist/core/abstractions/providers/config-parser.provider.spec.js +18 -0
- package/dist/core/abstractions/providers/config-parser.provider.spec.js.map +1 -0
- package/dist/core/abstractions/providers/logger.provider.spec.d.ts +1 -0
- package/dist/core/abstractions/providers/logger.provider.spec.js +212 -0
- package/dist/core/abstractions/providers/logger.provider.spec.js.map +1 -0
- package/dist/core/abstractions/providers/mock-filesystem.provider.d.ts +1 -0
- package/dist/core/abstractions/providers/mock-filesystem.provider.js +4 -0
- package/dist/core/abstractions/providers/mock-filesystem.provider.js.map +1 -1
- package/dist/core/abstractions/providers/mock-filesystem.provider.spec.d.ts +1 -0
- package/dist/core/abstractions/providers/mock-filesystem.provider.spec.js +56 -0
- package/dist/core/abstractions/providers/mock-filesystem.provider.spec.js.map +1 -0
- package/dist/core/abstractions/providers/node-filesystem.provider.d.ts +1 -0
- package/dist/core/abstractions/providers/node-filesystem.provider.js +4 -0
- package/dist/core/abstractions/providers/node-filesystem.provider.js.map +1 -1
- package/dist/core/architecture/nx-workspace.strategy.d.ts +3 -0
- package/dist/core/architecture/nx-workspace.strategy.js +35 -6
- package/dist/core/architecture/nx-workspace.strategy.js.map +1 -1
- package/dist/core/architecture/nx-workspace.strategy.spec.d.ts +1 -0
- package/dist/core/architecture/nx-workspace.strategy.spec.js +187 -0
- package/dist/core/architecture/nx-workspace.strategy.spec.js.map +1 -0
- package/dist/core/architecture/workspace-manager.strategy.d.ts +1 -0
- package/dist/core/di/container.spec.js +164 -0
- package/dist/core/di/container.spec.js.map +1 -1
- package/dist/core/generators/hexagonal-scaffolder.d.ts +6 -0
- package/dist/core/generators/hexagonal-scaffolder.js +260 -0
- package/dist/core/generators/hexagonal-scaffolder.js.map +1 -0
- package/dist/core/generators/hexagonal-scaffolder.spec.d.ts +1 -0
- package/dist/core/generators/hexagonal-scaffolder.spec.js +164 -0
- package/dist/core/generators/hexagonal-scaffolder.spec.js.map +1 -0
- package/dist/core/generators/mermaid-class-parser.d.ts +38 -0
- package/dist/core/generators/mermaid-class-parser.js +183 -0
- package/dist/core/generators/mermaid-class-parser.js.map +1 -0
- package/dist/core/generators/mermaid-class-parser.spec.d.ts +1 -0
- package/dist/core/generators/mermaid-class-parser.spec.js +148 -0
- package/dist/core/generators/mermaid-class-parser.spec.js.map +1 -0
- package/dist/core/mcp/metrics.service.spec.d.ts +1 -0
- package/dist/core/mcp/metrics.service.spec.js +159 -0
- package/dist/core/mcp/metrics.service.spec.js.map +1 -0
- package/dist/core/mcp/prompts/index.js +36 -0
- package/dist/core/mcp/prompts/index.js.map +1 -1
- package/dist/core/mcp/prompts/index.spec.d.ts +1 -0
- package/dist/core/mcp/prompts/index.spec.js +150 -0
- package/dist/core/mcp/prompts/index.spec.js.map +1 -0
- package/dist/core/mcp/resources/index.js +16 -0
- package/dist/core/mcp/resources/index.js.map +1 -1
- package/dist/core/mcp/resources/index.spec.d.ts +1 -0
- package/dist/core/mcp/resources/index.spec.js +212 -0
- package/dist/core/mcp/resources/index.spec.js.map +1 -0
- package/dist/core/mcp/server.d.ts +4 -1
- package/dist/core/mcp/server.js +250 -25
- package/dist/core/mcp/server.js.map +1 -1
- package/dist/core/mcp/server.spec.d.ts +1 -0
- package/dist/core/mcp/server.spec.js +579 -0
- package/dist/core/mcp/server.spec.js.map +1 -0
- package/dist/core/mcp/tools/agent.spec.d.ts +1 -0
- package/dist/core/mcp/tools/agent.spec.js +171 -0
- package/dist/core/mcp/tools/agent.spec.js.map +1 -0
- package/dist/core/mcp/tools/architecture.d.ts +2 -0
- package/dist/core/mcp/tools/architecture.js +68 -0
- package/dist/core/mcp/tools/architecture.js.map +1 -1
- package/dist/core/mcp/tools/architecture.spec.d.ts +1 -0
- package/dist/core/mcp/tools/architecture.spec.js +145 -0
- package/dist/core/mcp/tools/architecture.spec.js.map +1 -0
- package/dist/core/mcp/tools/gate.d.ts +1 -0
- package/dist/core/mcp/tools/gate.js +68 -0
- package/dist/core/mcp/tools/gate.js.map +1 -0
- package/dist/core/mcp/tools/moscow.d.ts +30 -0
- package/dist/core/mcp/tools/moscow.js +113 -0
- package/dist/core/mcp/tools/moscow.js.map +1 -0
- package/dist/core/mcp/tools/moscow.spec.d.ts +1 -0
- package/dist/core/mcp/tools/moscow.spec.js +209 -0
- package/dist/core/mcp/tools/moscow.spec.js.map +1 -0
- package/dist/core/mcp/tools/sdlc.js +2 -1
- package/dist/core/mcp/tools/sdlc.js.map +1 -1
- package/dist/core/mcp/tools/sdlc.spec.d.ts +1 -0
- package/dist/core/mcp/tools/sdlc.spec.js +170 -0
- package/dist/core/mcp/tools/sdlc.spec.js.map +1 -0
- package/dist/core/mcp/tools/validate.spec.d.ts +1 -0
- package/dist/core/mcp/tools/validate.spec.js +130 -0
- package/dist/core/mcp/tools/validate.spec.js.map +1 -0
- package/dist/core/mcp/watcher.service.spec.js +109 -8
- package/dist/core/mcp/watcher.service.spec.js.map +1 -1
- package/dist/core/metrics/dora-calculator.d.ts +18 -0
- package/dist/core/metrics/dora-calculator.js +146 -0
- package/dist/core/metrics/dora-calculator.js.map +1 -0
- package/dist/core/metrics/dora-calculator.spec.d.ts +1 -0
- package/dist/core/metrics/dora-calculator.spec.js +164 -0
- package/dist/core/metrics/dora-calculator.spec.js.map +1 -0
- package/dist/core/metrics/git-log-reader.d.ts +14 -0
- package/dist/core/metrics/git-log-reader.js +39 -0
- package/dist/core/metrics/git-log-reader.js.map +1 -0
- package/dist/core/metrics/git-log-reader.spec.d.ts +1 -0
- package/dist/core/metrics/git-log-reader.spec.js +108 -0
- package/dist/core/metrics/git-log-reader.spec.js.map +1 -0
- package/dist/core/observability/command-watcher.spec.d.ts +1 -0
- package/dist/core/observability/command-watcher.spec.js +123 -0
- package/dist/core/observability/command-watcher.spec.js.map +1 -0
- package/dist/core/observability/error-reporter.spec.d.ts +1 -0
- package/dist/core/observability/error-reporter.spec.js +291 -0
- package/dist/core/observability/error-reporter.spec.js.map +1 -0
- package/dist/core/observability/index.d.ts +1 -0
- package/dist/core/observability/index.js +3 -1
- package/dist/core/observability/index.js.map +1 -1
- package/dist/core/observability/structured-logger.spec.d.ts +1 -0
- package/dist/core/observability/structured-logger.spec.js +197 -0
- package/dist/core/observability/structured-logger.spec.js.map +1 -0
- package/dist/core/observability/timing.spec.d.ts +1 -0
- package/dist/core/observability/timing.spec.js +216 -0
- package/dist/core/observability/timing.spec.js.map +1 -0
- package/dist/core/observability/tool-usage-telemetry.service.d.ts +46 -0
- package/dist/core/observability/tool-usage-telemetry.service.js +181 -0
- package/dist/core/observability/tool-usage-telemetry.service.js.map +1 -0
- package/dist/core/services/command-history.service.d.ts +1 -0
- package/dist/core/services/command-history.service.js +4 -0
- package/dist/core/services/command-history.service.js.map +1 -1
- package/dist/core/services/command-history.service.spec.js +217 -126
- package/dist/core/services/command-history.service.spec.js.map +1 -1
- package/dist/core/upgrade/satellite-upgrade.service.spec.js +212 -132
- package/dist/core/upgrade/satellite-upgrade.service.spec.js.map +1 -1
- package/dist/core/validators/deep-architecture-analyzer.d.ts +58 -0
- package/dist/core/validators/deep-architecture-analyzer.js +333 -0
- package/dist/core/validators/deep-architecture-analyzer.js.map +1 -0
- package/dist/core/validators/deep-architecture-analyzer.spec.d.ts +1 -0
- package/dist/core/validators/deep-architecture-analyzer.spec.js +186 -0
- package/dist/core/validators/deep-architecture-analyzer.spec.js.map +1 -0
- package/dist/core/validators/evaluators/evaluator.interface.d.ts +14 -0
- package/dist/core/validators/evaluators/evaluator.interface.js +3 -0
- package/dist/core/validators/evaluators/evaluator.interface.js.map +1 -0
- package/dist/core/validators/evaluators/native-evaluator.d.ts +31 -0
- package/dist/core/validators/evaluators/native-evaluator.js +461 -0
- package/dist/core/validators/evaluators/native-evaluator.js.map +1 -0
- package/dist/core/validators/evaluators/opa-evaluator.d.ts +20 -0
- package/dist/core/validators/evaluators/opa-evaluator.js +249 -0
- package/dist/core/validators/evaluators/opa-evaluator.js.map +1 -0
- package/dist/core/validators/phase-gate-validator.service.d.ts +4 -0
- package/dist/core/validators/phase-gate-validator.service.js +76 -9
- package/dist/core/validators/phase-gate-validator.service.js.map +1 -1
- package/dist/core/validators/rule-evaluation-engine.d.ts +34 -0
- package/dist/core/validators/rule-evaluation-engine.js +187 -0
- package/dist/core/validators/rule-evaluation-engine.js.map +1 -0
- package/dist/core/validators/rule-evaluation-engine.spec.d.ts +1 -0
- package/dist/core/validators/rule-evaluation-engine.spec.js +243 -0
- package/dist/core/validators/rule-evaluation-engine.spec.js.map +1 -0
- package/dist/core/validators/ruleset-validator-architecture.spec.js +2 -2
- package/dist/core/validators/ruleset-validator-architecture.spec.js.map +1 -1
- package/dist/core/validators/ruleset-validator.service.d.ts +2 -0
- package/dist/core/validators/ruleset-validator.service.js +36 -11
- package/dist/core/validators/ruleset-validator.service.js.map +1 -1
- package/dist/core/validators/ruleset-validator.service.spec.js +347 -0
- package/dist/core/validators/ruleset-validator.service.spec.js.map +1 -1
- package/dist/domain/gate-evidence.d.ts +59 -0
- package/dist/domain/gate-evidence.js +38 -0
- package/dist/domain/gate-evidence.js.map +1 -0
- package/dist/domain/services/adr.service.d.ts +2 -2
- package/dist/domain/services/adr.service.js +12 -8
- package/dist/domain/services/adr.service.js.map +1 -1
- package/dist/domain/services/index.d.ts +1 -0
- package/dist/domain/services/index.js +3 -1
- package/dist/domain/services/index.js.map +1 -1
- package/dist/domain/services/moscow-prioritization.service.d.ts +44 -0
- package/dist/domain/services/moscow-prioritization.service.js +213 -0
- package/dist/domain/services/moscow-prioritization.service.js.map +1 -0
- package/dist/domain/services/moscow-prioritization.service.spec.d.ts +1 -0
- package/dist/domain/services/moscow-prioritization.service.spec.js +285 -0
- package/dist/domain/services/moscow-prioritization.service.spec.js.map +1 -0
- package/dist/domain/services/standards.service.spec.js +461 -1
- package/dist/domain/services/standards.service.spec.js.map +1 -1
- package/dist/infrastructure/adapters/webhook.adapter.d.ts +5 -0
- package/dist/infrastructure/adapters/webhook.adapter.js +19 -0
- package/dist/infrastructure/adapters/webhook.adapter.js.map +1 -0
- package/dist/infrastructure/cli/command-executor.spec.d.ts +1 -0
- package/dist/infrastructure/cli/command-executor.spec.js +149 -0
- package/dist/infrastructure/cli/command-executor.spec.js.map +1 -0
- package/dist/infrastructure/cli/providers/providers.spec.d.ts +1 -0
- package/dist/infrastructure/cli/providers/providers.spec.js +251 -0
- package/dist/infrastructure/cli/providers/providers.spec.js.map +1 -0
- package/package.json +23 -5
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const validate_1 = require("./validate");
|
|
4
|
+
jest.mock('../../validators/ruleset-validator.service', () => ({
|
|
5
|
+
RulesetValidatorService: jest.fn().mockImplementation(() => ({
|
|
6
|
+
validate: jest.fn(),
|
|
7
|
+
loadRulesetById: jest.fn(),
|
|
8
|
+
})),
|
|
9
|
+
}));
|
|
10
|
+
const ruleset_validator_service_1 = require("../../validators/ruleset-validator.service");
|
|
11
|
+
describe('MCP Tools - validate', () => {
|
|
12
|
+
let mockValidator;
|
|
13
|
+
beforeEach(() => {
|
|
14
|
+
jest.clearAllMocks();
|
|
15
|
+
mockValidator = new ruleset_validator_service_1.RulesetValidatorService();
|
|
16
|
+
});
|
|
17
|
+
describe('handleValidateTool', () => {
|
|
18
|
+
it('should return error when path is missing', async () => {
|
|
19
|
+
const result = await (0, validate_1.handleValidateTool)({}, mockValidator);
|
|
20
|
+
expect(result).toHaveProperty('error', true);
|
|
21
|
+
expect(result).toHaveProperty('message', 'path is required');
|
|
22
|
+
});
|
|
23
|
+
it('should validate repository when path provided', async () => {
|
|
24
|
+
mockValidator.validate.mockResolvedValue({
|
|
25
|
+
status: 'passed',
|
|
26
|
+
rulesChecked: 10,
|
|
27
|
+
issues: [],
|
|
28
|
+
});
|
|
29
|
+
const result = await (0, validate_1.handleValidateTool)({ path: '/test/repo' }, mockValidator);
|
|
30
|
+
expect(mockValidator.validate).toHaveBeenCalledWith('/test/repo', undefined);
|
|
31
|
+
expect(result).toHaveProperty('status', 'passed');
|
|
32
|
+
});
|
|
33
|
+
it('should pass corePath to validator', async () => {
|
|
34
|
+
mockValidator.validate.mockResolvedValue({
|
|
35
|
+
status: 'passed',
|
|
36
|
+
rulesChecked: 5,
|
|
37
|
+
issues: [],
|
|
38
|
+
});
|
|
39
|
+
await (0, validate_1.handleValidateTool)({ path: '/test/repo', corePath: '/core' }, mockValidator);
|
|
40
|
+
expect(mockValidator.validate).toHaveBeenCalledWith('/test/repo', '/core');
|
|
41
|
+
});
|
|
42
|
+
it('should return summary format when requested', async () => {
|
|
43
|
+
mockValidator.validate.mockResolvedValue({
|
|
44
|
+
status: 'passed',
|
|
45
|
+
rulesChecked: 10,
|
|
46
|
+
issues: [],
|
|
47
|
+
});
|
|
48
|
+
const result = await (0, validate_1.handleValidateTool)({ path: '/test/repo', format: 'summary' }, mockValidator);
|
|
49
|
+
expect(typeof result).toBe('string');
|
|
50
|
+
expect(result).toContain('PASSED');
|
|
51
|
+
});
|
|
52
|
+
it('should return table format when requested', async () => {
|
|
53
|
+
mockValidator.validate.mockResolvedValue({
|
|
54
|
+
status: 'failed',
|
|
55
|
+
rulesChecked: 10,
|
|
56
|
+
issues: [
|
|
57
|
+
{ ruleId: 'GOV-01', severity: 'MUST', title: 'Missing evolith.yaml', blocking: true, category: 'Governance' },
|
|
58
|
+
],
|
|
59
|
+
});
|
|
60
|
+
const result = await (0, validate_1.handleValidateTool)({ path: '/test/repo', format: 'table' }, mockValidator);
|
|
61
|
+
expect(typeof result).toBe('string');
|
|
62
|
+
expect(result).toContain('| Rule |');
|
|
63
|
+
expect(result).toContain('GOV-01');
|
|
64
|
+
});
|
|
65
|
+
it('should return JSON format by default', async () => {
|
|
66
|
+
mockValidator.validate.mockResolvedValue({
|
|
67
|
+
status: 'passed',
|
|
68
|
+
rulesChecked: 10,
|
|
69
|
+
issues: [],
|
|
70
|
+
});
|
|
71
|
+
const result = await (0, validate_1.handleValidateTool)({ path: '/test/repo' }, mockValidator);
|
|
72
|
+
expect(typeof result).not.toBe('string');
|
|
73
|
+
expect(result).toHaveProperty('status');
|
|
74
|
+
});
|
|
75
|
+
it('should load ruleset by ID when ruleset parameter provided', async () => {
|
|
76
|
+
mockValidator.loadRulesetById.mockResolvedValue([{ id: 'R-01' }]);
|
|
77
|
+
const result = await (0, validate_1.handleValidateTool)({
|
|
78
|
+
path: '/test/repo',
|
|
79
|
+
ruleset: 'governance',
|
|
80
|
+
corePath: '/core',
|
|
81
|
+
}, mockValidator);
|
|
82
|
+
expect(mockValidator.loadRulesetById).toHaveBeenCalledWith('/core', 'governance');
|
|
83
|
+
expect(result).toHaveProperty('ruleset', 'governance');
|
|
84
|
+
expect(result).toHaveProperty('issues');
|
|
85
|
+
});
|
|
86
|
+
it('should find core path automatically when not provided', async () => {
|
|
87
|
+
mockValidator.loadRulesetById.mockResolvedValue([]);
|
|
88
|
+
const result = await (0, validate_1.handleValidateTool)({
|
|
89
|
+
path: '/test/repo',
|
|
90
|
+
ruleset: 'governance',
|
|
91
|
+
}, mockValidator);
|
|
92
|
+
expect(mockValidator.loadRulesetById).toHaveBeenCalled();
|
|
93
|
+
expect(result).toHaveProperty('corePath');
|
|
94
|
+
});
|
|
95
|
+
it('should include timestamp in ruleset response', async () => {
|
|
96
|
+
mockValidator.loadRulesetById.mockResolvedValue([]);
|
|
97
|
+
const result = await (0, validate_1.handleValidateTool)({
|
|
98
|
+
path: '/test/repo',
|
|
99
|
+
ruleset: 'governance',
|
|
100
|
+
}, mockValidator);
|
|
101
|
+
expect(result).toHaveProperty('timestamp');
|
|
102
|
+
});
|
|
103
|
+
it('should format summary with issue count', async () => {
|
|
104
|
+
mockValidator.validate.mockResolvedValue({
|
|
105
|
+
status: 'failed',
|
|
106
|
+
rulesChecked: 15,
|
|
107
|
+
issues: [
|
|
108
|
+
{ ruleId: 'GOV-01', severity: 'MUST', title: 'Issue 1', blocking: true },
|
|
109
|
+
{ ruleId: 'GOV-02', severity: 'SHOULD', title: 'Issue 2', blocking: false },
|
|
110
|
+
],
|
|
111
|
+
});
|
|
112
|
+
const result = await (0, validate_1.handleValidateTool)({ path: '/test/repo', format: 'summary' }, mockValidator);
|
|
113
|
+
expect(result).toContain('Issues: 2');
|
|
114
|
+
});
|
|
115
|
+
it('should format table with blocking indicator', async () => {
|
|
116
|
+
mockValidator.validate.mockResolvedValue({
|
|
117
|
+
status: 'failed',
|
|
118
|
+
rulesChecked: 10,
|
|
119
|
+
issues: [
|
|
120
|
+
{ ruleId: 'GOV-01', severity: 'MUST', title: 'Blocking issue', blocking: true, category: 'Governance' },
|
|
121
|
+
{ ruleId: 'GOV-02', severity: 'SHOULD', title: 'Non-blocking', blocking: false, category: 'Governance' },
|
|
122
|
+
],
|
|
123
|
+
});
|
|
124
|
+
const result = await (0, validate_1.handleValidateTool)({ path: '/test/repo', format: 'table' }, mockValidator);
|
|
125
|
+
expect(result).toContain('YES');
|
|
126
|
+
expect(result).toContain('no');
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
});
|
|
130
|
+
//# sourceMappingURL=validate.spec.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validate.spec.js","sourceRoot":"","sources":["../../../../src/core/mcp/tools/validate.spec.ts"],"names":[],"mappings":"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"}
|
|
@@ -47,24 +47,125 @@ jest.mock('chokidar', () => {
|
|
|
47
47
|
});
|
|
48
48
|
describe('WatcherService', () => {
|
|
49
49
|
let service;
|
|
50
|
+
let watchMock;
|
|
50
51
|
beforeEach(async () => {
|
|
52
|
+
jest.clearAllMocks();
|
|
51
53
|
const module = await testing_1.Test.createTestingModule({
|
|
52
54
|
providers: [watcher_service_1.WatcherService],
|
|
53
55
|
}).compile();
|
|
54
56
|
service = module.get(watcher_service_1.WatcherService);
|
|
57
|
+
watchMock = {
|
|
58
|
+
on: jest.fn().mockReturnThis(),
|
|
59
|
+
close: jest.fn(),
|
|
60
|
+
};
|
|
61
|
+
chokidar.watch.mockReturnValue(watchMock);
|
|
55
62
|
});
|
|
56
63
|
it('should be defined', () => {
|
|
57
64
|
expect(service).toBeDefined();
|
|
58
65
|
});
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
66
|
+
describe('startWatching', () => {
|
|
67
|
+
it('should start watching the specified directory', () => {
|
|
68
|
+
service.startWatching('/test/cwd');
|
|
69
|
+
expect(chokidar.watch).toHaveBeenCalled();
|
|
70
|
+
});
|
|
71
|
+
it('should watch markdown files', () => {
|
|
72
|
+
service.startWatching('/test/cwd');
|
|
73
|
+
const callArgs = chokidar.watch.mock.calls[0];
|
|
74
|
+
const patterns = callArgs[0];
|
|
75
|
+
expect(patterns).toContain('**/*.md');
|
|
76
|
+
});
|
|
77
|
+
it('should watch package.json', () => {
|
|
78
|
+
service.startWatching('/test/cwd');
|
|
79
|
+
const callArgs = chokidar.watch.mock.calls[0];
|
|
80
|
+
const patterns = callArgs[0];
|
|
81
|
+
expect(patterns).toContain('package.json');
|
|
82
|
+
});
|
|
83
|
+
it('should watch evolith.setup.json', () => {
|
|
84
|
+
service.startWatching('/test/cwd');
|
|
85
|
+
const callArgs = chokidar.watch.mock.calls[0];
|
|
86
|
+
const patterns = callArgs[0];
|
|
87
|
+
expect(patterns).toContain('evolith.setup.json');
|
|
88
|
+
});
|
|
89
|
+
it('should use provided cwd in options', () => {
|
|
90
|
+
service.startWatching('/custom/path');
|
|
91
|
+
const callArgs = chokidar.watch.mock.calls[0];
|
|
92
|
+
const options = callArgs[1];
|
|
93
|
+
expect(options.cwd).toBe('/custom/path');
|
|
94
|
+
});
|
|
95
|
+
it('should use process.cwd() when no cwd is provided', () => {
|
|
96
|
+
service.startWatching();
|
|
97
|
+
const callArgs = chokidar.watch.mock.calls[0];
|
|
98
|
+
const options = callArgs[1];
|
|
99
|
+
expect(options.cwd).toBe(process.cwd());
|
|
100
|
+
});
|
|
101
|
+
it('should set persistent option to true', () => {
|
|
102
|
+
service.startWatching('/test/cwd');
|
|
103
|
+
const callArgs = chokidar.watch.mock.calls[0];
|
|
104
|
+
const options = callArgs[1];
|
|
105
|
+
expect(options.persistent).toBe(true);
|
|
106
|
+
});
|
|
107
|
+
it('should register change event handler', () => {
|
|
108
|
+
service.startWatching('/test/cwd');
|
|
109
|
+
expect(watchMock.on).toHaveBeenCalledWith('change', expect.any(Function));
|
|
110
|
+
});
|
|
111
|
+
it('should register error event handler', () => {
|
|
112
|
+
service.startWatching('/test/cwd');
|
|
113
|
+
expect(watchMock.on).toHaveBeenCalledWith('error', expect.any(Function));
|
|
114
|
+
});
|
|
115
|
+
it('should chain on calls', () => {
|
|
116
|
+
service.startWatching('/test/cwd');
|
|
117
|
+
expect(watchMock.on).toHaveBeenCalledTimes(2);
|
|
118
|
+
});
|
|
119
|
+
});
|
|
120
|
+
describe('handleFileChange', () => {
|
|
121
|
+
it('should trigger change handler when file changes', () => {
|
|
122
|
+
service.startWatching('/test/cwd');
|
|
123
|
+
const changeHandler = watchMock.on.mock.calls.find((call) => call[0] === 'change')?.[1];
|
|
124
|
+
expect(changeHandler).toBeDefined();
|
|
125
|
+
changeHandler('test-file.md');
|
|
126
|
+
});
|
|
127
|
+
it('should log architecture file changes', () => {
|
|
128
|
+
const logSpy = jest.spyOn(service['logger'], 'log').mockImplementation(() => { });
|
|
129
|
+
service.startWatching('/test/cwd');
|
|
130
|
+
const changeHandler = watchMock.on.mock.calls.find((call) => call[0] === 'change')?.[1];
|
|
131
|
+
changeHandler('architecture/pattern.md');
|
|
132
|
+
expect(logSpy).toHaveBeenCalledWith(expect.stringContaining('[IDE NOTIFY]'));
|
|
133
|
+
logSpy.mockRestore();
|
|
134
|
+
});
|
|
135
|
+
it('should log docs file changes', () => {
|
|
136
|
+
const logSpy = jest.spyOn(service['logger'], 'log').mockImplementation(() => { });
|
|
137
|
+
service.startWatching('/test/cwd');
|
|
138
|
+
const changeHandler = watchMock.on.mock.calls.find((call) => call[0] === 'change')?.[1];
|
|
139
|
+
changeHandler('docs/guide.md');
|
|
140
|
+
expect(logSpy).toHaveBeenCalledWith(expect.stringContaining('[IDE NOTIFY]'));
|
|
141
|
+
logSpy.mockRestore();
|
|
142
|
+
});
|
|
143
|
+
it('should not notify for non-relevant files', () => {
|
|
144
|
+
const logSpy = jest.spyOn(service['logger'], 'log').mockImplementation(() => { });
|
|
145
|
+
service.startWatching('/test/cwd');
|
|
146
|
+
const changeHandler = watchMock.on.mock.calls.find((call) => call[0] === 'change')?.[1];
|
|
147
|
+
changeHandler('src/index.ts');
|
|
148
|
+
const notifyCalls = logSpy.mock.calls.filter((call) => call[0]?.includes?.('[IDE NOTIFY]'));
|
|
149
|
+
expect(notifyCalls).toHaveLength(0);
|
|
150
|
+
logSpy.mockRestore();
|
|
151
|
+
});
|
|
62
152
|
});
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
153
|
+
describe('onModuleDestroy', () => {
|
|
154
|
+
it('should close the watcher on module destroy', () => {
|
|
155
|
+
service.startWatching('/test/cwd');
|
|
156
|
+
service.onModuleDestroy();
|
|
157
|
+
expect(watchMock.close).toHaveBeenCalled();
|
|
158
|
+
});
|
|
159
|
+
it('should not throw when watcher is not started', () => {
|
|
160
|
+
expect(() => service.onModuleDestroy()).not.toThrow();
|
|
161
|
+
});
|
|
162
|
+
it('should log when watcher is stopped', () => {
|
|
163
|
+
const logSpy = jest.spyOn(service['logger'], 'log').mockImplementation(() => { });
|
|
164
|
+
service.startWatching('/test/cwd');
|
|
165
|
+
service.onModuleDestroy();
|
|
166
|
+
expect(logSpy).toHaveBeenCalledWith('Watcher detenido.');
|
|
167
|
+
logSpy.mockRestore();
|
|
168
|
+
});
|
|
68
169
|
});
|
|
69
170
|
});
|
|
70
171
|
//# sourceMappingURL=watcher.service.spec.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"watcher.service.spec.js","sourceRoot":"","sources":["../../../src/core/mcp/watcher.service.spec.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,6CAAsD;AACtD,uDAAmD;AACnD,mDAAqC;AAErC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,GAAG,EAAE;IACzB,MAAM,WAAW,GAAG;QAClB,EAAE,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,cAAc,EAAE;QAC9B,KAAK,EAAE,IAAI,CAAC,EAAE,EAAE;KACjB,CAAC;IACF,OAAO;QACL,KAAK,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,WAAW,CAAC;KAClC,CAAC;AACJ,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;IAC9B,IAAI,OAAuB,CAAC;
|
|
1
|
+
{"version":3,"file":"watcher.service.spec.js","sourceRoot":"","sources":["../../../src/core/mcp/watcher.service.spec.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { GitCommit } from './git-log-reader';
|
|
2
|
+
export type DoraRating = 'elite' | 'high' | 'medium' | 'low' | 'unknown';
|
|
3
|
+
export interface DoraMetric {
|
|
4
|
+
value: number;
|
|
5
|
+
label: string;
|
|
6
|
+
rating: DoraRating;
|
|
7
|
+
unit: string;
|
|
8
|
+
}
|
|
9
|
+
export interface DoraMetrics {
|
|
10
|
+
deploymentFrequency: DoraMetric;
|
|
11
|
+
leadTimeForChanges: DoraMetric;
|
|
12
|
+
changeFailureRate: DoraMetric;
|
|
13
|
+
timeToRestore: DoraMetric;
|
|
14
|
+
analyzedDays: number;
|
|
15
|
+
totalCommits: number;
|
|
16
|
+
available: boolean;
|
|
17
|
+
}
|
|
18
|
+
export declare function calculateDora(commits: GitCommit[], sinceDays: number): DoraMetrics;
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.calculateDora = calculateDora;
|
|
4
|
+
function rateDeployFreq(perWeek) {
|
|
5
|
+
if (perWeek > 7)
|
|
6
|
+
return 'elite';
|
|
7
|
+
if (perWeek >= 1)
|
|
8
|
+
return 'high';
|
|
9
|
+
if (perWeek >= 0.25)
|
|
10
|
+
return 'medium';
|
|
11
|
+
return 'low';
|
|
12
|
+
}
|
|
13
|
+
function rateLeadTime(hours) {
|
|
14
|
+
if (hours < 1)
|
|
15
|
+
return 'elite';
|
|
16
|
+
if (hours < 168)
|
|
17
|
+
return 'high';
|
|
18
|
+
if (hours < 720)
|
|
19
|
+
return 'medium';
|
|
20
|
+
return 'low';
|
|
21
|
+
}
|
|
22
|
+
function rateFailureRate(pct) {
|
|
23
|
+
if (pct <= 5)
|
|
24
|
+
return 'elite';
|
|
25
|
+
if (pct <= 10)
|
|
26
|
+
return 'high';
|
|
27
|
+
if (pct <= 15)
|
|
28
|
+
return 'medium';
|
|
29
|
+
return 'low';
|
|
30
|
+
}
|
|
31
|
+
function rateMttr(hours) {
|
|
32
|
+
if (hours < 1)
|
|
33
|
+
return 'elite';
|
|
34
|
+
if (hours < 24)
|
|
35
|
+
return 'high';
|
|
36
|
+
if (hours < 168)
|
|
37
|
+
return 'medium';
|
|
38
|
+
return 'low';
|
|
39
|
+
}
|
|
40
|
+
function median(values) {
|
|
41
|
+
if (values.length === 0)
|
|
42
|
+
return 0;
|
|
43
|
+
const sorted = [...values].sort((a, b) => a - b);
|
|
44
|
+
const mid = Math.floor(sorted.length / 2);
|
|
45
|
+
return sorted.length % 2 === 0
|
|
46
|
+
? (sorted[mid - 1] + sorted[mid]) / 2
|
|
47
|
+
: sorted[mid];
|
|
48
|
+
}
|
|
49
|
+
function hoursLabel(hours) {
|
|
50
|
+
if (hours < 1)
|
|
51
|
+
return `${Math.round(hours * 60)} min`;
|
|
52
|
+
if (hours < 24)
|
|
53
|
+
return `${hours.toFixed(1)} h`;
|
|
54
|
+
return `${(hours / 24).toFixed(1)} days`;
|
|
55
|
+
}
|
|
56
|
+
const FIX_RE = /^(fix|hotfix|revert|rollback|bugfix|patch)[\s!(:]/i;
|
|
57
|
+
function calculateDora(commits, sinceDays) {
|
|
58
|
+
if (commits.length === 0) {
|
|
59
|
+
return emptyMetrics(sinceDays);
|
|
60
|
+
}
|
|
61
|
+
const weeks = sinceDays / 7;
|
|
62
|
+
const deploys = commits.filter(c => c.isMerge);
|
|
63
|
+
const deployCount = deploys.length > 0 ? deploys.length : commits.length;
|
|
64
|
+
const deploysPerWeek = deployCount / weeks;
|
|
65
|
+
const deployFreq = {
|
|
66
|
+
value: deploysPerWeek,
|
|
67
|
+
label: deploysPerWeek >= 1
|
|
68
|
+
? `${deploysPerWeek.toFixed(1)}/week`
|
|
69
|
+
: `${(deploysPerWeek * 4).toFixed(1)}/month`,
|
|
70
|
+
rating: rateDeployFreq(deploysPerWeek),
|
|
71
|
+
unit: 'deployments/week',
|
|
72
|
+
};
|
|
73
|
+
const commitDateMs = new Map();
|
|
74
|
+
for (const c of commits) {
|
|
75
|
+
commitDateMs.set(c.hash, new Date(c.date).getTime());
|
|
76
|
+
}
|
|
77
|
+
const leadTimesHours = [];
|
|
78
|
+
for (const merge of deploys) {
|
|
79
|
+
const mergeMs = commitDateMs.get(merge.hash) ?? 0;
|
|
80
|
+
const featureHash = merge.parents[1];
|
|
81
|
+
const featureMs = featureHash ? (commitDateMs.get(featureHash) ?? mergeMs) : mergeMs;
|
|
82
|
+
const diffHours = (mergeMs - featureMs) / 3_600_000;
|
|
83
|
+
if (diffHours > 0 && diffHours < 24 * 30) {
|
|
84
|
+
leadTimesHours.push(diffHours);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
const medianLeadHours = leadTimesHours.length > 0 ? median(leadTimesHours) : 0;
|
|
88
|
+
const leadTime = {
|
|
89
|
+
value: medianLeadHours,
|
|
90
|
+
label: leadTimesHours.length > 0 ? hoursLabel(medianLeadHours) : 'n/a (no merges)',
|
|
91
|
+
rating: leadTimesHours.length > 0 ? rateLeadTime(medianLeadHours) : 'unknown',
|
|
92
|
+
unit: 'hours (median)',
|
|
93
|
+
};
|
|
94
|
+
const fixCommits = commits.filter(c => FIX_RE.test(c.subject));
|
|
95
|
+
const cfr = (fixCommits.length / commits.length) * 100;
|
|
96
|
+
const changeFailureRate = {
|
|
97
|
+
value: cfr,
|
|
98
|
+
label: `${cfr.toFixed(1)}%`,
|
|
99
|
+
rating: rateFailureRate(cfr),
|
|
100
|
+
unit: '% fix/revert commits',
|
|
101
|
+
};
|
|
102
|
+
const sorted = [...commits].sort((a, b) => new Date(a.date).getTime() - new Date(b.date).getTime());
|
|
103
|
+
const restoreTimesHours = [];
|
|
104
|
+
for (let i = 0; i < sorted.length; i++) {
|
|
105
|
+
if (!FIX_RE.test(sorted[i].subject))
|
|
106
|
+
continue;
|
|
107
|
+
for (let j = i - 1; j >= 0; j--) {
|
|
108
|
+
if (!FIX_RE.test(sorted[j].subject)) {
|
|
109
|
+
const diffH = (new Date(sorted[i].date).getTime() - new Date(sorted[j].date).getTime()) / 3_600_000;
|
|
110
|
+
if (diffH > 0 && diffH < 24 * 14) {
|
|
111
|
+
restoreTimesHours.push(diffH);
|
|
112
|
+
}
|
|
113
|
+
break;
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
const medianRestoreHours = restoreTimesHours.length > 0 ? median(restoreTimesHours) : 0;
|
|
118
|
+
const timeToRestore = {
|
|
119
|
+
value: medianRestoreHours,
|
|
120
|
+
label: restoreTimesHours.length > 0 ? hoursLabel(medianRestoreHours) : 'n/a (no fix commits)',
|
|
121
|
+
rating: restoreTimesHours.length > 0 ? rateMttr(medianRestoreHours) : 'unknown',
|
|
122
|
+
unit: 'hours (median)',
|
|
123
|
+
};
|
|
124
|
+
return {
|
|
125
|
+
deploymentFrequency: deployFreq,
|
|
126
|
+
leadTimeForChanges: leadTime,
|
|
127
|
+
changeFailureRate,
|
|
128
|
+
timeToRestore,
|
|
129
|
+
analyzedDays: sinceDays,
|
|
130
|
+
totalCommits: commits.length,
|
|
131
|
+
available: true,
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
function emptyMetrics(sinceDays) {
|
|
135
|
+
const none = { value: 0, label: 'n/a', rating: 'unknown', unit: '' };
|
|
136
|
+
return {
|
|
137
|
+
deploymentFrequency: none,
|
|
138
|
+
leadTimeForChanges: none,
|
|
139
|
+
changeFailureRate: none,
|
|
140
|
+
timeToRestore: none,
|
|
141
|
+
analyzedDays: sinceDays,
|
|
142
|
+
totalCommits: 0,
|
|
143
|
+
available: false,
|
|
144
|
+
};
|
|
145
|
+
}
|
|
146
|
+
//# sourceMappingURL=dora-calculator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"dora-calculator.js","sourceRoot":"","sources":["../../../src/core/metrics/dora-calculator.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const dora_calculator_1 = require("./dora-calculator");
|
|
4
|
+
function makeCommit(overrides) {
|
|
5
|
+
return {
|
|
6
|
+
subject: 'feat: some feature',
|
|
7
|
+
parents: ['abc'],
|
|
8
|
+
isMerge: false,
|
|
9
|
+
...overrides,
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
function dailyCommits(count, startDaysAgo = 89) {
|
|
13
|
+
const now = Date.now();
|
|
14
|
+
return Array.from({ length: count }, (_, i) => {
|
|
15
|
+
const ms = now - (startDaysAgo - i) * 86_400_000;
|
|
16
|
+
return makeCommit({ hash: `hash${i}`, date: new Date(ms).toISOString() });
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
describe('calculateDora', () => {
|
|
20
|
+
describe('empty history', () => {
|
|
21
|
+
it('returns available=false when no commits', () => {
|
|
22
|
+
const result = (0, dora_calculator_1.calculateDora)([], 90);
|
|
23
|
+
expect(result.available).toBe(false);
|
|
24
|
+
expect(result.totalCommits).toBe(0);
|
|
25
|
+
});
|
|
26
|
+
it('all metrics return unknown rating', () => {
|
|
27
|
+
const result = (0, dora_calculator_1.calculateDora)([], 90);
|
|
28
|
+
expect(result.deploymentFrequency.rating).toBe('unknown');
|
|
29
|
+
expect(result.leadTimeForChanges.rating).toBe('unknown');
|
|
30
|
+
expect(result.changeFailureRate.rating).toBe('unknown');
|
|
31
|
+
expect(result.timeToRestore.rating).toBe('unknown');
|
|
32
|
+
});
|
|
33
|
+
});
|
|
34
|
+
describe('deployment frequency', () => {
|
|
35
|
+
it('rates daily commits as high (≥1/week)', () => {
|
|
36
|
+
const commits = dailyCommits(90);
|
|
37
|
+
const result = (0, dora_calculator_1.calculateDora)(commits, 90);
|
|
38
|
+
expect(result.deploymentFrequency.rating).toBe('high');
|
|
39
|
+
expect(result.deploymentFrequency.value).toBeGreaterThan(1);
|
|
40
|
+
});
|
|
41
|
+
it('rates multiple-per-day as elite (>7/week)', () => {
|
|
42
|
+
const commits = Array.from({ length: 200 }, (_, i) => makeCommit({ hash: `h${i}`, date: new Date(Date.now() - i * 3_600_000).toISOString() }));
|
|
43
|
+
const result = (0, dora_calculator_1.calculateDora)(commits, 7);
|
|
44
|
+
expect(result.deploymentFrequency.rating).toBe('elite');
|
|
45
|
+
});
|
|
46
|
+
it('prefers merge commits over all commits', () => {
|
|
47
|
+
const commits = dailyCommits(90);
|
|
48
|
+
const merges = [0, 30, 60].map(i => makeCommit({
|
|
49
|
+
hash: `merge${i}`,
|
|
50
|
+
date: commits[i].date,
|
|
51
|
+
isMerge: true,
|
|
52
|
+
parents: [commits[i].hash, `feature${i}`],
|
|
53
|
+
subject: `Merge branch 'feature${i}'`,
|
|
54
|
+
}));
|
|
55
|
+
const result = (0, dora_calculator_1.calculateDora)([...commits, ...merges], 90);
|
|
56
|
+
expect(result.deploymentFrequency.rating).toBe('low');
|
|
57
|
+
expect(result.deploymentFrequency.value).toBeCloseTo(3 / (90 / 7), 1);
|
|
58
|
+
});
|
|
59
|
+
it('rates 2 deploys per week as high', () => {
|
|
60
|
+
const merges = Array.from({ length: 26 }, (_, i) => makeCommit({
|
|
61
|
+
hash: `m${i}`,
|
|
62
|
+
date: new Date(Date.now() - i * 3 * 86_400_000).toISOString(),
|
|
63
|
+
isMerge: true,
|
|
64
|
+
parents: ['p1', `f${i}`],
|
|
65
|
+
subject: 'Merge branch feature',
|
|
66
|
+
}));
|
|
67
|
+
const result = (0, dora_calculator_1.calculateDora)(merges, 90);
|
|
68
|
+
expect(result.deploymentFrequency.rating).toBe('high');
|
|
69
|
+
});
|
|
70
|
+
});
|
|
71
|
+
describe('change failure rate', () => {
|
|
72
|
+
it('returns 0% when no fix commits', () => {
|
|
73
|
+
const result = (0, dora_calculator_1.calculateDora)(dailyCommits(10), 90);
|
|
74
|
+
expect(result.changeFailureRate.value).toBe(0);
|
|
75
|
+
expect(result.changeFailureRate.rating).toBe('elite');
|
|
76
|
+
});
|
|
77
|
+
it('detects fix: prefix', () => {
|
|
78
|
+
const commits = [
|
|
79
|
+
makeCommit({ hash: 'h1', date: new Date().toISOString(), subject: 'feat: new thing' }),
|
|
80
|
+
makeCommit({ hash: 'h2', date: new Date().toISOString(), subject: 'fix: broken thing' }),
|
|
81
|
+
makeCommit({ hash: 'h3', date: new Date().toISOString(), subject: 'fix: another bug' }),
|
|
82
|
+
makeCommit({ hash: 'h4', date: new Date().toISOString(), subject: 'feat: another feature' }),
|
|
83
|
+
];
|
|
84
|
+
const result = (0, dora_calculator_1.calculateDora)(commits, 90);
|
|
85
|
+
expect(result.changeFailureRate.value).toBe(50);
|
|
86
|
+
});
|
|
87
|
+
it('detects hotfix, revert, rollback prefixes', () => {
|
|
88
|
+
const commits = [
|
|
89
|
+
makeCommit({ hash: 'h1', date: new Date().toISOString(), subject: 'hotfix: urgent' }),
|
|
90
|
+
makeCommit({ hash: 'h2', date: new Date().toISOString(), subject: 'revert: bad deploy' }),
|
|
91
|
+
makeCommit({ hash: 'h3', date: new Date().toISOString(), subject: 'rollback: db migration' }),
|
|
92
|
+
makeCommit({ hash: 'h4', date: new Date().toISOString(), subject: 'feat: clean feature' }),
|
|
93
|
+
];
|
|
94
|
+
const result = (0, dora_calculator_1.calculateDora)(commits, 90);
|
|
95
|
+
expect(result.changeFailureRate.value).toBe(75);
|
|
96
|
+
expect(result.changeFailureRate.rating).toBe('low');
|
|
97
|
+
});
|
|
98
|
+
it('rates 3% as elite', () => {
|
|
99
|
+
const commits = [
|
|
100
|
+
...Array.from({ length: 97 }, (_, i) => makeCommit({ hash: `feat${i}`, date: new Date().toISOString() })),
|
|
101
|
+
...Array.from({ length: 3 }, (_, i) => makeCommit({ hash: `fix${i}`, date: new Date().toISOString(), subject: 'fix: bug' })),
|
|
102
|
+
];
|
|
103
|
+
const result = (0, dora_calculator_1.calculateDora)(commits, 90);
|
|
104
|
+
expect(result.changeFailureRate.rating).toBe('elite');
|
|
105
|
+
});
|
|
106
|
+
});
|
|
107
|
+
describe('lead time for changes', () => {
|
|
108
|
+
it('returns unknown when no merge commits', () => {
|
|
109
|
+
const result = (0, dora_calculator_1.calculateDora)(dailyCommits(10), 90);
|
|
110
|
+
expect(result.leadTimeForChanges.rating).toBe('unknown');
|
|
111
|
+
});
|
|
112
|
+
it('calculates lead time from merge parent timestamps', () => {
|
|
113
|
+
const now = Date.now();
|
|
114
|
+
const featureDate = new Date(now - 2 * 3_600_000).toISOString();
|
|
115
|
+
const mergeDate = new Date(now).toISOString();
|
|
116
|
+
const feature = makeCommit({ hash: 'feature1', date: featureDate });
|
|
117
|
+
const merge = makeCommit({
|
|
118
|
+
hash: 'merge1',
|
|
119
|
+
date: mergeDate,
|
|
120
|
+
isMerge: true,
|
|
121
|
+
parents: ['main', 'feature1'],
|
|
122
|
+
subject: "Merge branch 'feature'",
|
|
123
|
+
});
|
|
124
|
+
const result = (0, dora_calculator_1.calculateDora)([feature, merge], 90);
|
|
125
|
+
expect(result.leadTimeForChanges.value).toBeCloseTo(2, 0);
|
|
126
|
+
expect(result.leadTimeForChanges.rating).toBe('high');
|
|
127
|
+
});
|
|
128
|
+
it('rates sub-hour lead time as elite', () => {
|
|
129
|
+
const now = Date.now();
|
|
130
|
+
const feature = makeCommit({ hash: 'f1', date: new Date(now - 30 * 60_000).toISOString() });
|
|
131
|
+
const merge = makeCommit({
|
|
132
|
+
hash: 'm1', date: new Date(now).toISOString(),
|
|
133
|
+
isMerge: true, parents: ['main', 'f1'],
|
|
134
|
+
subject: 'Merge',
|
|
135
|
+
});
|
|
136
|
+
const result = (0, dora_calculator_1.calculateDora)([feature, merge], 90);
|
|
137
|
+
expect(result.leadTimeForChanges.rating).toBe('elite');
|
|
138
|
+
});
|
|
139
|
+
});
|
|
140
|
+
describe('time to restore', () => {
|
|
141
|
+
it('returns unknown when no fix commits', () => {
|
|
142
|
+
const result = (0, dora_calculator_1.calculateDora)(dailyCommits(10), 90);
|
|
143
|
+
expect(result.timeToRestore.rating).toBe('unknown');
|
|
144
|
+
});
|
|
145
|
+
it('calculates time between bug introduction and fix', () => {
|
|
146
|
+
const now = Date.now();
|
|
147
|
+
const feat = makeCommit({ hash: 'feat1', date: new Date(now - 4 * 3_600_000).toISOString(), subject: 'feat: deploy' });
|
|
148
|
+
const fix = makeCommit({ hash: 'fix1', date: new Date(now).toISOString(), subject: 'fix: resolve production issue' });
|
|
149
|
+
const result = (0, dora_calculator_1.calculateDora)([feat, fix], 90);
|
|
150
|
+
expect(result.timeToRestore.value).toBeCloseTo(4, 0);
|
|
151
|
+
expect(result.timeToRestore.rating).toBe('high');
|
|
152
|
+
});
|
|
153
|
+
});
|
|
154
|
+
describe('metadata', () => {
|
|
155
|
+
it('reports correct analyzedDays and totalCommits', () => {
|
|
156
|
+
const commits = dailyCommits(30);
|
|
157
|
+
const result = (0, dora_calculator_1.calculateDora)(commits, 90);
|
|
158
|
+
expect(result.analyzedDays).toBe(90);
|
|
159
|
+
expect(result.totalCommits).toBe(30);
|
|
160
|
+
expect(result.available).toBe(true);
|
|
161
|
+
});
|
|
162
|
+
});
|
|
163
|
+
});
|
|
164
|
+
//# sourceMappingURL=dora-calculator.spec.js.map
|