@wundr.io/cli 1.0.12 → 1.0.13
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/LICENSE +21 -0
- package/dist/ai/ai-service.d.ts +152 -0
- package/dist/ai/ai-service.d.ts.map +1 -0
- package/dist/ai/ai-service.js +430 -0
- package/dist/ai/ai-service.js.map +1 -0
- package/dist/ai/claude-client.d.ts +130 -0
- package/dist/ai/claude-client.d.ts.map +1 -0
- package/dist/ai/claude-client.js +340 -0
- package/dist/ai/claude-client.js.map +1 -0
- package/dist/ai/conversation-manager.d.ts +164 -0
- package/dist/ai/conversation-manager.d.ts.map +1 -0
- package/dist/ai/conversation-manager.js +614 -0
- package/dist/ai/conversation-manager.js.map +1 -0
- package/dist/ai/index.d.ts +5 -0
- package/dist/ai/index.d.ts.map +1 -0
- package/dist/ai/index.js +8 -0
- package/dist/ai/index.js.map +1 -0
- package/dist/cli.d.ts +36 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +192 -0
- package/dist/cli.js.map +1 -0
- package/dist/commands/ai.d.ts +89 -0
- package/dist/commands/ai.d.ts.map +1 -0
- package/dist/commands/ai.js +954 -0
- package/dist/commands/ai.js.map +1 -0
- package/dist/commands/alignment.d.ts +78 -0
- package/dist/commands/alignment.d.ts.map +1 -0
- package/dist/commands/alignment.js +817 -0
- package/dist/commands/alignment.js.map +1 -0
- package/dist/commands/analyze-optimized.d.ts +14 -0
- package/dist/commands/analyze-optimized.d.ts.map +1 -0
- package/dist/commands/analyze-optimized.js +609 -0
- package/dist/commands/analyze-optimized.js.map +1 -0
- package/dist/commands/analyze.d.ts +65 -0
- package/dist/commands/analyze.d.ts.map +1 -0
- package/dist/commands/analyze.js +435 -0
- package/dist/commands/analyze.js.map +1 -0
- package/dist/commands/batch.d.ts +93 -0
- package/dist/commands/batch.d.ts.map +1 -0
- package/dist/commands/batch.js +854 -0
- package/dist/commands/batch.js.map +1 -0
- package/dist/commands/chat.d.ts +72 -0
- package/dist/commands/chat.d.ts.map +1 -0
- package/dist/commands/chat.js +678 -0
- package/dist/commands/chat.js.map +1 -0
- package/dist/commands/claude-init.d.ts +28 -0
- package/dist/commands/claude-init.d.ts.map +1 -0
- package/dist/commands/claude-init.js +591 -0
- package/dist/commands/claude-init.js.map +1 -0
- package/dist/commands/claude-setup.d.ts +119 -0
- package/dist/commands/claude-setup.d.ts.map +1 -0
- package/dist/commands/claude-setup.js +1079 -0
- package/dist/commands/claude-setup.js.map +1 -0
- package/dist/commands/computer-setup.d.ts +8 -0
- package/dist/commands/computer-setup.d.ts.map +1 -0
- package/dist/commands/computer-setup.js +877 -0
- package/dist/commands/computer-setup.js.map +1 -0
- package/dist/commands/create-command.d.ts +7 -0
- package/dist/commands/create-command.d.ts.map +1 -0
- package/dist/commands/create-command.js +158 -0
- package/dist/commands/create-command.js.map +1 -0
- package/dist/commands/create.d.ts +74 -0
- package/dist/commands/create.d.ts.map +1 -0
- package/dist/commands/create.js +556 -0
- package/dist/commands/create.js.map +1 -0
- package/dist/commands/dashboard.d.ts +91 -0
- package/dist/commands/dashboard.d.ts.map +1 -0
- package/dist/commands/dashboard.js +538 -0
- package/dist/commands/dashboard.js.map +1 -0
- package/dist/commands/govern.d.ts +70 -0
- package/dist/commands/govern.d.ts.map +1 -0
- package/dist/commands/govern.js +481 -0
- package/dist/commands/govern.js.map +1 -0
- package/dist/commands/governance.d.ts +17 -0
- package/dist/commands/governance.d.ts.map +1 -0
- package/dist/commands/governance.js +703 -0
- package/dist/commands/governance.js.map +1 -0
- package/dist/commands/guardian.d.ts +20 -0
- package/dist/commands/guardian.d.ts.map +1 -0
- package/dist/commands/guardian.js +597 -0
- package/dist/commands/guardian.js.map +1 -0
- package/dist/commands/init.d.ts +59 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +650 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/orchestrator.d.ts +7 -0
- package/dist/commands/orchestrator.d.ts.map +1 -0
- package/dist/commands/orchestrator.js +578 -0
- package/dist/commands/orchestrator.js.map +1 -0
- package/dist/commands/performance-optimizer.d.ts +30 -0
- package/dist/commands/performance-optimizer.d.ts.map +1 -0
- package/dist/commands/performance-optimizer.js +650 -0
- package/dist/commands/performance-optimizer.js.map +1 -0
- package/dist/commands/plugins.d.ts +87 -0
- package/dist/commands/plugins.d.ts.map +1 -0
- package/dist/commands/plugins.js +685 -0
- package/dist/commands/plugins.js.map +1 -0
- package/dist/commands/rag.d.ts +7 -0
- package/dist/commands/rag.d.ts.map +1 -0
- package/dist/commands/rag.js +751 -0
- package/dist/commands/rag.js.map +1 -0
- package/dist/commands/session.d.ts +41 -0
- package/dist/commands/session.d.ts.map +1 -0
- package/dist/commands/session.js +441 -0
- package/dist/commands/session.js.map +1 -0
- package/dist/commands/setup.d.ts +24 -0
- package/dist/commands/setup.d.ts.map +1 -0
- package/dist/commands/setup.js +172 -0
- package/dist/commands/setup.js.map +1 -0
- package/dist/commands/test-init.d.ts +9 -0
- package/dist/commands/test-init.d.ts.map +1 -0
- package/dist/commands/test-init.js +222 -0
- package/dist/commands/test-init.js.map +1 -0
- package/dist/commands/test.d.ts +25 -0
- package/dist/commands/test.d.ts.map +1 -0
- package/dist/commands/test.js +217 -0
- package/dist/commands/test.js.map +1 -0
- package/dist/commands/watch.d.ts +76 -0
- package/dist/commands/watch.d.ts.map +1 -0
- package/dist/commands/watch.js +613 -0
- package/dist/commands/watch.js.map +1 -0
- package/dist/commands/worktree.d.ts +63 -0
- package/dist/commands/worktree.d.ts.map +1 -0
- package/dist/commands/worktree.js +774 -0
- package/dist/commands/worktree.js.map +1 -0
- package/dist/context/context-manager.d.ts +155 -0
- package/dist/context/context-manager.d.ts.map +1 -0
- package/dist/context/context-manager.js +383 -0
- package/dist/context/context-manager.js.map +1 -0
- package/dist/context/index.d.ts +3 -0
- package/dist/context/index.d.ts.map +1 -0
- package/dist/context/index.js +6 -0
- package/dist/context/index.js.map +1 -0
- package/dist/context/session-manager.d.ts +207 -0
- package/dist/context/session-manager.d.ts.map +1 -0
- package/dist/context/session-manager.js +686 -0
- package/dist/context/session-manager.js.map +1 -0
- package/dist/framework/command-interface.d.ts +349 -0
- package/dist/framework/command-interface.d.ts.map +1 -0
- package/dist/framework/command-interface.js +101 -0
- package/dist/framework/command-interface.js.map +1 -0
- package/dist/framework/command-registry.d.ts +173 -0
- package/dist/framework/command-registry.d.ts.map +1 -0
- package/dist/framework/command-registry.js +734 -0
- package/dist/framework/command-registry.js.map +1 -0
- package/dist/framework/completion-exporter.d.ts +79 -0
- package/dist/framework/completion-exporter.d.ts.map +1 -0
- package/dist/framework/completion-exporter.js +259 -0
- package/dist/framework/completion-exporter.js.map +1 -0
- package/dist/framework/debug-logger.d.ts +163 -0
- package/dist/framework/debug-logger.d.ts.map +1 -0
- package/dist/framework/debug-logger.js +373 -0
- package/dist/framework/debug-logger.js.map +1 -0
- package/dist/framework/error-handler.d.ts +196 -0
- package/dist/framework/error-handler.d.ts.map +1 -0
- package/dist/framework/error-handler.js +613 -0
- package/dist/framework/error-handler.js.map +1 -0
- package/dist/framework/help-generator.d.ts +78 -0
- package/dist/framework/help-generator.d.ts.map +1 -0
- package/dist/framework/help-generator.js +414 -0
- package/dist/framework/help-generator.js.map +1 -0
- package/dist/framework/index.d.ts +62 -0
- package/dist/framework/index.d.ts.map +1 -0
- package/dist/framework/index.js +95 -0
- package/dist/framework/index.js.map +1 -0
- package/dist/framework/interactive-repl.d.ts +138 -0
- package/dist/framework/interactive-repl.d.ts.map +1 -0
- package/dist/framework/interactive-repl.js +567 -0
- package/dist/framework/interactive-repl.js.map +1 -0
- package/dist/framework/output-formatter.d.ts +274 -0
- package/dist/framework/output-formatter.d.ts.map +1 -0
- package/dist/framework/output-formatter.js +545 -0
- package/dist/framework/output-formatter.js.map +1 -0
- package/dist/framework/progress-manager.d.ts +192 -0
- package/dist/framework/progress-manager.d.ts.map +1 -0
- package/dist/framework/progress-manager.js +408 -0
- package/dist/framework/progress-manager.js.map +1 -0
- package/dist/index.d.ts +8 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +51 -0
- package/dist/index.js.map +1 -0
- package/dist/interactive/interactive-mode.d.ts +76 -0
- package/dist/interactive/interactive-mode.d.ts.map +1 -0
- package/dist/interactive/interactive-mode.js +732 -0
- package/dist/interactive/interactive-mode.js.map +1 -0
- package/dist/nlp/command-mapper.d.ts +174 -0
- package/dist/nlp/command-mapper.d.ts.map +1 -0
- package/dist/nlp/command-mapper.js +624 -0
- package/dist/nlp/command-mapper.js.map +1 -0
- package/dist/nlp/command-parser.d.ts +106 -0
- package/dist/nlp/command-parser.d.ts.map +1 -0
- package/dist/nlp/command-parser.js +417 -0
- package/dist/nlp/command-parser.js.map +1 -0
- package/dist/nlp/index.d.ts +5 -0
- package/dist/nlp/index.d.ts.map +1 -0
- package/dist/nlp/index.js +8 -0
- package/dist/nlp/index.js.map +1 -0
- package/dist/nlp/intent-classifier.d.ts +59 -0
- package/dist/nlp/intent-classifier.d.ts.map +1 -0
- package/dist/nlp/intent-classifier.js +384 -0
- package/dist/nlp/intent-classifier.js.map +1 -0
- package/dist/nlp/intent-parser.d.ts +152 -0
- package/dist/nlp/intent-parser.d.ts.map +1 -0
- package/dist/nlp/intent-parser.js +746 -0
- package/dist/nlp/intent-parser.js.map +1 -0
- package/dist/plugins/plugin-manager.d.ts +121 -0
- package/dist/plugins/plugin-manager.d.ts.map +1 -0
- package/dist/plugins/plugin-manager.js +606 -0
- package/dist/plugins/plugin-manager.js.map +1 -0
- package/dist/types/index.d.ts +224 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +3 -0
- package/dist/types/index.js.map +1 -0
- package/dist/utils/backup-rollback-manager.d.ts +72 -0
- package/dist/utils/backup-rollback-manager.d.ts.map +1 -0
- package/dist/utils/backup-rollback-manager.js +288 -0
- package/dist/utils/backup-rollback-manager.js.map +1 -0
- package/dist/utils/claude-config-installer.d.ts +98 -0
- package/dist/utils/claude-config-installer.d.ts.map +1 -0
- package/dist/utils/claude-config-installer.js +678 -0
- package/dist/utils/claude-config-installer.js.map +1 -0
- package/dist/utils/config-manager.d.ts +73 -0
- package/dist/utils/config-manager.d.ts.map +1 -0
- package/dist/utils/config-manager.js +339 -0
- package/dist/utils/config-manager.js.map +1 -0
- package/dist/utils/error-handler.d.ts +46 -0
- package/dist/utils/error-handler.d.ts.map +1 -0
- package/dist/utils/error-handler.js +169 -0
- package/dist/utils/error-handler.js.map +1 -0
- package/dist/utils/logger.d.ts +25 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +105 -0
- package/dist/utils/logger.js.map +1 -0
- package/package.json +23 -23
|
@@ -0,0 +1,703 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Governance CLI Commands - IPRE Compliance
|
|
4
|
+
*
|
|
5
|
+
* Implements governance commands from Appendix B:
|
|
6
|
+
* - wundr governance check - Run IPRE compliance check
|
|
7
|
+
* - wundr governance report - Generate alignment debt report
|
|
8
|
+
* - wundr governance status - Show current governance status
|
|
9
|
+
* - wundr governance validate <file> - Validate IPRE config file
|
|
10
|
+
*/
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.createAlignmentCommand = void 0;
|
|
13
|
+
exports.createGovernanceCommand = createGovernanceCommand;
|
|
14
|
+
exports.governanceCommand = createGovernanceCommand;
|
|
15
|
+
const tslib_1 = require("tslib");
|
|
16
|
+
const path_1 = tslib_1.__importDefault(require("path"));
|
|
17
|
+
const chalk_1 = tslib_1.__importDefault(require("chalk"));
|
|
18
|
+
const commander_1 = require("commander");
|
|
19
|
+
const fs_extra_1 = tslib_1.__importDefault(require("fs-extra"));
|
|
20
|
+
const yaml_1 = tslib_1.__importDefault(require("yaml"));
|
|
21
|
+
const governance_1 = require("@wundr.io/governance");
|
|
22
|
+
const error_handler_1 = require("../utils/error-handler");
|
|
23
|
+
const logger_1 = require("../utils/logger");
|
|
24
|
+
// ============================================================================
|
|
25
|
+
// Governance Command Factory
|
|
26
|
+
// ============================================================================
|
|
27
|
+
/**
|
|
28
|
+
* Creates the governance command with all subcommands
|
|
29
|
+
*/
|
|
30
|
+
function createGovernanceCommand() {
|
|
31
|
+
const governance = new commander_1.Command('governance')
|
|
32
|
+
.alias('gov')
|
|
33
|
+
.description('IPRE governance and compliance tools');
|
|
34
|
+
// Check subcommand
|
|
35
|
+
governance
|
|
36
|
+
.command('check')
|
|
37
|
+
.description('Run IPRE compliance check')
|
|
38
|
+
.option('-s, --session <id>', 'Session ID to check')
|
|
39
|
+
.option('-v, --verbose', 'Enable verbose output')
|
|
40
|
+
.action(async (options) => {
|
|
41
|
+
await runComplianceCheck(options);
|
|
42
|
+
});
|
|
43
|
+
// Report subcommand
|
|
44
|
+
governance
|
|
45
|
+
.command('report')
|
|
46
|
+
.description('Generate alignment debt report')
|
|
47
|
+
.option('-s, --session <id>', 'Session ID for the report')
|
|
48
|
+
.option('-o, --output <file>', 'Output file path')
|
|
49
|
+
.option('-f, --format <format>', 'Output format (md|json)', 'md')
|
|
50
|
+
.action(async (options) => {
|
|
51
|
+
await generateAlignmentReport(options);
|
|
52
|
+
});
|
|
53
|
+
// Status subcommand
|
|
54
|
+
governance
|
|
55
|
+
.command('status')
|
|
56
|
+
.description('Show current governance status')
|
|
57
|
+
.action(async () => {
|
|
58
|
+
await showGovernanceStatus();
|
|
59
|
+
});
|
|
60
|
+
// Validate subcommand
|
|
61
|
+
governance
|
|
62
|
+
.command('validate <file>')
|
|
63
|
+
.description('Validate IPRE config file')
|
|
64
|
+
.action(async (file) => {
|
|
65
|
+
await validateIPREConfig(file);
|
|
66
|
+
});
|
|
67
|
+
return governance;
|
|
68
|
+
}
|
|
69
|
+
// Re-export the full-featured alignment command from the dedicated module
|
|
70
|
+
// This provides: report, score, history, and dimensions subcommands
|
|
71
|
+
var alignment_1 = require("./alignment");
|
|
72
|
+
Object.defineProperty(exports, "createAlignmentCommand", { enumerable: true, get: function () { return alignment_1.createAlignmentCommand; } });
|
|
73
|
+
// ============================================================================
|
|
74
|
+
// Command Implementations
|
|
75
|
+
// ============================================================================
|
|
76
|
+
/**
|
|
77
|
+
* Run IPRE compliance check
|
|
78
|
+
*/
|
|
79
|
+
async function runComplianceCheck(options) {
|
|
80
|
+
try {
|
|
81
|
+
logger_1.logger.info('Running IPRE compliance check...');
|
|
82
|
+
const sessionId = options.session || `session-${Date.now()}`;
|
|
83
|
+
const verbose = options.verbose || false;
|
|
84
|
+
// Create evaluation context
|
|
85
|
+
const context = {
|
|
86
|
+
evaluationId: `eval-${Date.now()}`,
|
|
87
|
+
timestamp: new Date(),
|
|
88
|
+
source: 'manual',
|
|
89
|
+
repository: process.cwd(),
|
|
90
|
+
metadata: {
|
|
91
|
+
sessionId,
|
|
92
|
+
},
|
|
93
|
+
};
|
|
94
|
+
// Create evaluator suite
|
|
95
|
+
const suite = (0, governance_1.createEvaluatorSuite)();
|
|
96
|
+
// Run policy compliance check
|
|
97
|
+
console.log(chalk_1.default.blue('\n--- Policy Compliance ---'));
|
|
98
|
+
const complianceResult = await suite.policyCompliance.checkPolicyCompliance(context);
|
|
99
|
+
displayComplianceResult(complianceResult, verbose);
|
|
100
|
+
// Run reward alignment check
|
|
101
|
+
console.log(chalk_1.default.blue('\n--- Reward Alignment ---'));
|
|
102
|
+
const alignmentEvaluator = (0, governance_1.createEvaluator)('reward_alignment');
|
|
103
|
+
const alignmentResult = await alignmentEvaluator.evaluate(context);
|
|
104
|
+
displayAlignmentResult(alignmentResult, verbose);
|
|
105
|
+
// Run drift detection
|
|
106
|
+
console.log(chalk_1.default.blue('\n--- Drift Detection ---'));
|
|
107
|
+
const driftEvaluator = (0, governance_1.createEvaluator)('drift_detection');
|
|
108
|
+
const driftResult = await driftEvaluator.evaluate(context);
|
|
109
|
+
displayDriftResult(driftResult, verbose);
|
|
110
|
+
// Run full suite and show summary
|
|
111
|
+
console.log(chalk_1.default.blue('\n--- Overall Summary ---'));
|
|
112
|
+
const evaluators = [
|
|
113
|
+
suite.policyCompliance,
|
|
114
|
+
suite.rewardAlignment,
|
|
115
|
+
suite.driftDetection,
|
|
116
|
+
];
|
|
117
|
+
const suiteResult = await (0, governance_1.runEvaluatorSuite)(evaluators, context);
|
|
118
|
+
displaySummary(suiteResult);
|
|
119
|
+
// Save state for future checks
|
|
120
|
+
await saveGovernanceState(sessionId, {
|
|
121
|
+
complianceResult,
|
|
122
|
+
alignmentResult,
|
|
123
|
+
driftResult,
|
|
124
|
+
suiteResult,
|
|
125
|
+
});
|
|
126
|
+
if (suiteResult.passed) {
|
|
127
|
+
logger_1.logger.success('IPRE compliance check passed');
|
|
128
|
+
}
|
|
129
|
+
else {
|
|
130
|
+
logger_1.logger.warn('IPRE compliance check found issues');
|
|
131
|
+
process.exitCode = 1;
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
catch (error) {
|
|
135
|
+
throw error_handler_1.errorHandler.createError('WUNDR_GOV_CHECK_FAILED', 'Failed to run IPRE compliance check', { options }, true);
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Generate alignment debt report
|
|
140
|
+
*/
|
|
141
|
+
async function generateAlignmentReport(options) {
|
|
142
|
+
try {
|
|
143
|
+
logger_1.logger.info('Generating alignment debt report...');
|
|
144
|
+
const sessionId = options.session || null;
|
|
145
|
+
const format = options.format || 'md';
|
|
146
|
+
const outputPath = options.output || `alignment-report-${Date.now()}.${format}`;
|
|
147
|
+
// Create evaluation context
|
|
148
|
+
const context = {
|
|
149
|
+
evaluationId: `eval-${Date.now()}`,
|
|
150
|
+
timestamp: new Date(),
|
|
151
|
+
source: 'manual',
|
|
152
|
+
repository: process.cwd(),
|
|
153
|
+
};
|
|
154
|
+
// Run all evaluations
|
|
155
|
+
const suite = (0, governance_1.createEvaluatorSuite)();
|
|
156
|
+
const complianceResult = await suite.policyCompliance.checkPolicyCompliance(context);
|
|
157
|
+
// Get alignment gaps
|
|
158
|
+
const alignmentEvaluator = (0, governance_1.createEvaluator)('reward_alignment');
|
|
159
|
+
const alignmentEvalResult = await alignmentEvaluator.evaluate(context);
|
|
160
|
+
// Get drift indicators
|
|
161
|
+
const driftEvaluator = (0, governance_1.createEvaluator)('drift_detection');
|
|
162
|
+
const driftEvalResult = await driftEvaluator.evaluate(context);
|
|
163
|
+
// Build report
|
|
164
|
+
const report = {
|
|
165
|
+
generatedAt: new Date().toISOString(),
|
|
166
|
+
sessionId,
|
|
167
|
+
summary: {
|
|
168
|
+
totalDebt: calculateAlignmentDebt(complianceResult, alignmentEvalResult, driftEvalResult),
|
|
169
|
+
criticalIssues: countCriticalIssues(complianceResult),
|
|
170
|
+
highPriorityItems: countHighPriorityItems(alignmentEvalResult),
|
|
171
|
+
alignmentScore: alignmentEvalResult.score,
|
|
172
|
+
complianceScore: complianceResult.score,
|
|
173
|
+
driftScore: 1 - driftEvalResult.score, // Invert since lower drift is better
|
|
174
|
+
},
|
|
175
|
+
policyViolations: complianceResult.violations.map(v => ({
|
|
176
|
+
policyId: v.policyId,
|
|
177
|
+
policyName: v.policyName,
|
|
178
|
+
severity: v.severity,
|
|
179
|
+
description: v.description,
|
|
180
|
+
location: v.location,
|
|
181
|
+
suggestedFix: v.suggestedFix,
|
|
182
|
+
})),
|
|
183
|
+
alignmentGaps: extractAlignmentGaps(alignmentEvalResult),
|
|
184
|
+
driftAlerts: extractDriftAlerts(driftEvalResult),
|
|
185
|
+
recommendations: generateRecommendations(complianceResult, alignmentEvalResult, driftEvalResult),
|
|
186
|
+
};
|
|
187
|
+
// Output report
|
|
188
|
+
if (format === 'json') {
|
|
189
|
+
await fs_extra_1.default.writeJson(outputPath, report, { spaces: 2 });
|
|
190
|
+
}
|
|
191
|
+
else {
|
|
192
|
+
const markdown = formatReportAsMarkdown(report);
|
|
193
|
+
await fs_extra_1.default.writeFile(outputPath, markdown, 'utf-8');
|
|
194
|
+
}
|
|
195
|
+
logger_1.logger.success(`Alignment debt report generated: ${outputPath}`);
|
|
196
|
+
// Display summary
|
|
197
|
+
console.log(chalk_1.default.blue('\n--- Report Summary ---'));
|
|
198
|
+
console.log(`Total Alignment Debt: ${chalk_1.default.yellow(report.summary.totalDebt.toFixed(2))}`);
|
|
199
|
+
console.log(`Critical Issues: ${chalk_1.default.red(report.summary.criticalIssues)}`);
|
|
200
|
+
console.log(`High Priority Items: ${chalk_1.default.yellow(report.summary.highPriorityItems)}`);
|
|
201
|
+
console.log(`Alignment Score: ${colorScore(report.summary.alignmentScore)}`);
|
|
202
|
+
console.log(`Compliance Score: ${colorScore(report.summary.complianceScore)}`);
|
|
203
|
+
console.log(`Drift Score: ${colorScore(1 - report.summary.driftScore)}`);
|
|
204
|
+
}
|
|
205
|
+
catch (error) {
|
|
206
|
+
throw error_handler_1.errorHandler.createError('WUNDR_GOV_REPORT_FAILED', 'Failed to generate alignment debt report', { options }, true);
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* Show current governance status
|
|
211
|
+
*/
|
|
212
|
+
async function showGovernanceStatus() {
|
|
213
|
+
try {
|
|
214
|
+
logger_1.logger.info('Fetching governance status...');
|
|
215
|
+
// Load saved state if available
|
|
216
|
+
const state = await loadGovernanceState();
|
|
217
|
+
// Create policy engine and get stats
|
|
218
|
+
const policyEngine = new governance_1.PolicyEngine();
|
|
219
|
+
const violationStats = policyEngine.getViolationStats();
|
|
220
|
+
const status = {
|
|
221
|
+
alignmentScore: state?.suiteResult?.overallScore ?? 0,
|
|
222
|
+
policyViolations: violationStats.total,
|
|
223
|
+
recentInterventions: await loadRecentInterventions(),
|
|
224
|
+
lastCheck: state?.timestamp ? new Date(state.timestamp) : null,
|
|
225
|
+
driftIndicators: extractDriftIndicatorsFromState(state),
|
|
226
|
+
};
|
|
227
|
+
// Display status
|
|
228
|
+
console.log(chalk_1.default.cyan('\n===================================='));
|
|
229
|
+
console.log(chalk_1.default.cyan(' GOVERNANCE STATUS'));
|
|
230
|
+
console.log(chalk_1.default.cyan('====================================\n'));
|
|
231
|
+
// Alignment Score
|
|
232
|
+
console.log(chalk_1.default.bold('Alignment Score:'));
|
|
233
|
+
displayProgressBar(status.alignmentScore);
|
|
234
|
+
console.log(` ${colorScore(status.alignmentScore)} (${(status.alignmentScore * 100).toFixed(1)}%)\n`);
|
|
235
|
+
// Policy Violations
|
|
236
|
+
console.log(chalk_1.default.bold('Policy Violations:'));
|
|
237
|
+
if (status.policyViolations === 0) {
|
|
238
|
+
console.log(chalk_1.default.green(' No active violations'));
|
|
239
|
+
}
|
|
240
|
+
else {
|
|
241
|
+
console.log(chalk_1.default.red(` ${status.policyViolations} active violations`));
|
|
242
|
+
console.log(` - Security: ${violationStats.byCategory.security}`);
|
|
243
|
+
console.log(` - Compliance: ${violationStats.byCategory.compliance}`);
|
|
244
|
+
console.log(` - Operational: ${violationStats.byCategory.operational}`);
|
|
245
|
+
}
|
|
246
|
+
console.log();
|
|
247
|
+
// Recent Interventions
|
|
248
|
+
console.log(chalk_1.default.bold('Recent Interventions:'));
|
|
249
|
+
if (status.recentInterventions.length === 0) {
|
|
250
|
+
console.log(chalk_1.default.green(' No recent interventions'));
|
|
251
|
+
}
|
|
252
|
+
else {
|
|
253
|
+
for (const intervention of status.recentInterventions.slice(0, 5)) {
|
|
254
|
+
const statusIcon = intervention.resolved
|
|
255
|
+
? chalk_1.default.green('[RESOLVED]')
|
|
256
|
+
: chalk_1.default.yellow('[OPEN]');
|
|
257
|
+
console.log(` ${statusIcon} ${intervention.type}: ${intervention.description}`);
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
console.log();
|
|
261
|
+
// Drift Indicators
|
|
262
|
+
console.log(chalk_1.default.bold('Drift Indicators:'));
|
|
263
|
+
if (status.driftIndicators.length === 0) {
|
|
264
|
+
console.log(chalk_1.default.green(' No significant drift detected'));
|
|
265
|
+
}
|
|
266
|
+
else {
|
|
267
|
+
for (const indicator of status.driftIndicators) {
|
|
268
|
+
const severityColor = getSeverityColor(indicator.severity);
|
|
269
|
+
const direction = indicator.change > 0 ? '+' : '';
|
|
270
|
+
console.log(` ${severityColor(`[${indicator.severity.toUpperCase()}]`)} ` +
|
|
271
|
+
`${indicator.pattern}: ${direction}${(indicator.change * 100).toFixed(1)}%`);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
console.log();
|
|
275
|
+
// Last Check
|
|
276
|
+
if (status.lastCheck) {
|
|
277
|
+
console.log(chalk_1.default.gray(`Last check: ${status.lastCheck.toLocaleString()}`));
|
|
278
|
+
}
|
|
279
|
+
else {
|
|
280
|
+
console.log(chalk_1.default.gray('No previous checks found. Run "wundr governance check" to start.'));
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
catch (error) {
|
|
284
|
+
throw error_handler_1.errorHandler.createError('WUNDR_GOV_STATUS_FAILED', 'Failed to fetch governance status', {}, true);
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
/**
|
|
288
|
+
* Validate IPRE config file
|
|
289
|
+
*/
|
|
290
|
+
async function validateIPREConfig(file) {
|
|
291
|
+
try {
|
|
292
|
+
const filePath = path_1.default.isAbsolute(file)
|
|
293
|
+
? file
|
|
294
|
+
: path_1.default.join(process.cwd(), file);
|
|
295
|
+
logger_1.logger.info(`Validating IPRE config: ${filePath}`);
|
|
296
|
+
// Check file exists
|
|
297
|
+
if (!(await fs_extra_1.default.pathExists(filePath))) {
|
|
298
|
+
logger_1.logger.error(`File not found: ${filePath}`);
|
|
299
|
+
process.exitCode = 1;
|
|
300
|
+
return;
|
|
301
|
+
}
|
|
302
|
+
// Read and parse file
|
|
303
|
+
const content = await fs_extra_1.default.readFile(filePath, 'utf-8');
|
|
304
|
+
let config;
|
|
305
|
+
const ext = path_1.default.extname(file).toLowerCase();
|
|
306
|
+
if (ext === '.yaml' || ext === '.yml') {
|
|
307
|
+
config = yaml_1.default.parse(content);
|
|
308
|
+
}
|
|
309
|
+
else if (ext === '.json') {
|
|
310
|
+
config = JSON.parse(content);
|
|
311
|
+
}
|
|
312
|
+
else {
|
|
313
|
+
logger_1.logger.error('Unsupported file format. Use .yaml, .yml, or .json');
|
|
314
|
+
process.exitCode = 1;
|
|
315
|
+
return;
|
|
316
|
+
}
|
|
317
|
+
// Validate schema
|
|
318
|
+
const validationErrors = validateIPRESchema(config);
|
|
319
|
+
if (validationErrors.length === 0) {
|
|
320
|
+
console.log(chalk_1.default.green('\nIPRE configuration is valid'));
|
|
321
|
+
displayConfigSummary(config);
|
|
322
|
+
}
|
|
323
|
+
else {
|
|
324
|
+
console.log(chalk_1.default.red('\nIPRE configuration validation failed:\n'));
|
|
325
|
+
for (const error of validationErrors) {
|
|
326
|
+
console.log(chalk_1.default.red(` - ${error}`));
|
|
327
|
+
}
|
|
328
|
+
process.exitCode = 1;
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
catch (error) {
|
|
332
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
333
|
+
throw error_handler_1.errorHandler.createError('WUNDR_GOV_VALIDATE_FAILED', `Failed to validate IPRE config: ${message}`, { file }, true);
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
// ============================================================================
|
|
337
|
+
// Helper Functions
|
|
338
|
+
// ============================================================================
|
|
339
|
+
function displayComplianceResult(result, verbose) {
|
|
340
|
+
const statusIcon = result.compliant
|
|
341
|
+
? chalk_1.default.green('[PASS]')
|
|
342
|
+
: chalk_1.default.red('[FAIL]');
|
|
343
|
+
console.log(`${statusIcon} Compliance Score: ${colorScore(result.score)}`);
|
|
344
|
+
console.log(` Passed Policies: ${result.passedPolicies.length}`);
|
|
345
|
+
console.log(` Skipped Policies: ${result.skippedPolicies.length}`);
|
|
346
|
+
console.log(` Violations: ${result.violations.length}`);
|
|
347
|
+
if (verbose && result.violations.length > 0) {
|
|
348
|
+
console.log(chalk_1.default.yellow('\n Violations:'));
|
|
349
|
+
for (const violation of result.violations) {
|
|
350
|
+
console.log(` - [${violation.severity.toUpperCase()}] ${violation.policyName}`);
|
|
351
|
+
console.log(` ${violation.description}`);
|
|
352
|
+
if (violation.suggestedFix) {
|
|
353
|
+
console.log(chalk_1.default.gray(` Fix: ${violation.suggestedFix}`));
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
}
|
|
358
|
+
function displayAlignmentResult(result, verbose) {
|
|
359
|
+
const statusIcon = result.passed
|
|
360
|
+
? chalk_1.default.green('[PASS]')
|
|
361
|
+
: chalk_1.default.red('[FAIL]');
|
|
362
|
+
console.log(`${statusIcon} Alignment Score: ${colorScore(result.score)}`);
|
|
363
|
+
console.log(` Issues: ${result.issues.length}`);
|
|
364
|
+
console.log(` Recommendations: ${result.recommendations.length}`);
|
|
365
|
+
if (verbose && result.issues.length > 0) {
|
|
366
|
+
console.log(chalk_1.default.yellow('\n Issues:'));
|
|
367
|
+
for (const issue of result.issues) {
|
|
368
|
+
console.log(` - ${issue}`);
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
if (verbose && result.recommendations.length > 0) {
|
|
372
|
+
console.log(chalk_1.default.blue('\n Recommendations:'));
|
|
373
|
+
for (const rec of result.recommendations) {
|
|
374
|
+
console.log(` - ${rec}`);
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
function displayDriftResult(result, verbose) {
|
|
379
|
+
// Note: For drift, higher score means less drift (inverted in evaluation)
|
|
380
|
+
const driftScore = 1 - result.score;
|
|
381
|
+
const statusIcon = result.passed
|
|
382
|
+
? chalk_1.default.green('[PASS]')
|
|
383
|
+
: chalk_1.default.red('[FAIL]');
|
|
384
|
+
console.log(`${statusIcon} Drift Score: ${colorScore(result.score)} (${(driftScore * 100).toFixed(1)}% drift)`);
|
|
385
|
+
console.log(` Drift Alerts: ${result.issues.length}`);
|
|
386
|
+
if (verbose && result.issues.length > 0) {
|
|
387
|
+
console.log(chalk_1.default.yellow('\n Drift Alerts:'));
|
|
388
|
+
for (const issue of result.issues) {
|
|
389
|
+
console.log(` - ${issue}`);
|
|
390
|
+
}
|
|
391
|
+
}
|
|
392
|
+
}
|
|
393
|
+
function displaySummary(result) {
|
|
394
|
+
const statusIcon = result.passed
|
|
395
|
+
? chalk_1.default.green('[PASSED]')
|
|
396
|
+
: chalk_1.default.red('[FAILED]');
|
|
397
|
+
console.log(`\n${statusIcon} Overall Score: ${colorScore(result.overallScore)}`);
|
|
398
|
+
if (result.criticalIssues.length > 0) {
|
|
399
|
+
console.log(chalk_1.default.red(`\nCritical Issues (${result.criticalIssues.length}):`));
|
|
400
|
+
for (const issue of result.criticalIssues.slice(0, 5)) {
|
|
401
|
+
console.log(chalk_1.default.red(` - ${issue}`));
|
|
402
|
+
}
|
|
403
|
+
if (result.criticalIssues.length > 5) {
|
|
404
|
+
console.log(chalk_1.default.gray(` ... and ${result.criticalIssues.length - 5} more`));
|
|
405
|
+
}
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
function colorScore(score) {
|
|
409
|
+
const percentage = (score * 100).toFixed(1);
|
|
410
|
+
if (score >= 0.9) {
|
|
411
|
+
return chalk_1.default.green(`${percentage}%`);
|
|
412
|
+
}
|
|
413
|
+
if (score >= 0.7) {
|
|
414
|
+
return chalk_1.default.yellow(`${percentage}%`);
|
|
415
|
+
}
|
|
416
|
+
return chalk_1.default.red(`${percentage}%`);
|
|
417
|
+
}
|
|
418
|
+
function getSeverityColor(severity) {
|
|
419
|
+
switch (severity) {
|
|
420
|
+
case 'critical':
|
|
421
|
+
return chalk_1.default.red;
|
|
422
|
+
case 'high':
|
|
423
|
+
return chalk_1.default.yellow;
|
|
424
|
+
case 'medium':
|
|
425
|
+
return chalk_1.default.blue;
|
|
426
|
+
default:
|
|
427
|
+
return chalk_1.default.gray;
|
|
428
|
+
}
|
|
429
|
+
}
|
|
430
|
+
function displayProgressBar(score) {
|
|
431
|
+
const width = 30;
|
|
432
|
+
const filled = Math.round(score * width);
|
|
433
|
+
const empty = width - filled;
|
|
434
|
+
const bar = chalk_1.default.green('[' + '='.repeat(filled)) + chalk_1.default.gray('-'.repeat(empty) + ']');
|
|
435
|
+
console.log(` ${bar}`);
|
|
436
|
+
}
|
|
437
|
+
function calculateAlignmentDebt(compliance, alignment, drift) {
|
|
438
|
+
// Calculate debt as weighted sum of issues
|
|
439
|
+
const complianceDebt = (1 - compliance.score) * 0.4;
|
|
440
|
+
const alignmentDebt = (1 - alignment.score) * 0.35;
|
|
441
|
+
const driftDebt = (1 - drift.score) * 0.25;
|
|
442
|
+
return complianceDebt + alignmentDebt + driftDebt;
|
|
443
|
+
}
|
|
444
|
+
function countCriticalIssues(compliance) {
|
|
445
|
+
return compliance.violations.filter(v => v.severity === 'critical').length;
|
|
446
|
+
}
|
|
447
|
+
function countHighPriorityItems(alignment) {
|
|
448
|
+
// Count issues that contain high-priority indicators
|
|
449
|
+
return alignment.issues.filter(issue => issue.toLowerCase().includes('critical') ||
|
|
450
|
+
issue.toLowerCase().includes('high')).length;
|
|
451
|
+
}
|
|
452
|
+
function extractAlignmentGaps(result) {
|
|
453
|
+
// Parse alignment gaps from issues
|
|
454
|
+
const gaps = [];
|
|
455
|
+
for (const issue of result.issues) {
|
|
456
|
+
const match = issue.match(/gap in (\w+): expected ([\d.]+), got ([\d.]+)/i);
|
|
457
|
+
if (match) {
|
|
458
|
+
const expected = parseFloat(match[2] ?? '0');
|
|
459
|
+
const actual = parseFloat(match[3] ?? '0');
|
|
460
|
+
gaps.push({
|
|
461
|
+
dimension: match[1] ?? 'unknown',
|
|
462
|
+
expected,
|
|
463
|
+
actual,
|
|
464
|
+
gap: expected - actual,
|
|
465
|
+
priority: expected - actual > 0.2
|
|
466
|
+
? 'high'
|
|
467
|
+
: expected - actual > 0.1
|
|
468
|
+
? 'medium'
|
|
469
|
+
: 'low',
|
|
470
|
+
});
|
|
471
|
+
}
|
|
472
|
+
}
|
|
473
|
+
return gaps;
|
|
474
|
+
}
|
|
475
|
+
function extractDriftAlerts(result) {
|
|
476
|
+
// Parse drift alerts from issues
|
|
477
|
+
const alerts = [];
|
|
478
|
+
for (const issue of result.issues) {
|
|
479
|
+
const match = issue.match(/Drift.*in (\w+): ([\d.]+)% (\w+)/i);
|
|
480
|
+
if (match) {
|
|
481
|
+
const changePercent = parseFloat(match[2] ?? '0');
|
|
482
|
+
alerts.push({
|
|
483
|
+
pattern: match[1] ?? 'unknown',
|
|
484
|
+
baselineValue: 0.9, // Default baseline
|
|
485
|
+
currentValue: 0.9 * (1 + changePercent / 100),
|
|
486
|
+
changePercent,
|
|
487
|
+
direction: match[3] ?? 'stable',
|
|
488
|
+
severity: changePercent >= 30
|
|
489
|
+
? 'critical'
|
|
490
|
+
: changePercent >= 20
|
|
491
|
+
? 'high'
|
|
492
|
+
: 'medium',
|
|
493
|
+
});
|
|
494
|
+
}
|
|
495
|
+
}
|
|
496
|
+
return alerts;
|
|
497
|
+
}
|
|
498
|
+
function generateRecommendations(compliance, alignment, drift) {
|
|
499
|
+
const recommendations = [];
|
|
500
|
+
// Add compliance recommendations
|
|
501
|
+
if (compliance.violations.length > 0) {
|
|
502
|
+
recommendations.push('Address policy violations immediately, especially critical ones');
|
|
503
|
+
const criticalCount = compliance.violations.filter(v => v.severity === 'critical').length;
|
|
504
|
+
if (criticalCount > 0) {
|
|
505
|
+
recommendations.push(`Fix ${criticalCount} critical policy violation(s) before deployment`);
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
// Add alignment recommendations from evaluator
|
|
509
|
+
recommendations.push(...alignment.recommendations);
|
|
510
|
+
// Add drift recommendations
|
|
511
|
+
if (drift.issues.length > 0) {
|
|
512
|
+
recommendations.push('Investigate drift causes and consider recalibration');
|
|
513
|
+
}
|
|
514
|
+
// General recommendations
|
|
515
|
+
if (compliance.score < 0.8) {
|
|
516
|
+
recommendations.push('Review and update security policies');
|
|
517
|
+
}
|
|
518
|
+
if (alignment.score < 0.85) {
|
|
519
|
+
recommendations.push('Schedule alignment review session');
|
|
520
|
+
}
|
|
521
|
+
return [...new Set(recommendations)]; // Remove duplicates
|
|
522
|
+
}
|
|
523
|
+
function formatReportAsMarkdown(report) {
|
|
524
|
+
let md = '# Alignment Debt Report\n\n';
|
|
525
|
+
md += `Generated: ${report.generatedAt}\n`;
|
|
526
|
+
if (report.sessionId) {
|
|
527
|
+
md += `Session: ${report.sessionId}\n`;
|
|
528
|
+
}
|
|
529
|
+
md += '\n---\n\n';
|
|
530
|
+
// Summary
|
|
531
|
+
md += '## Summary\n\n';
|
|
532
|
+
md += '| Metric | Value |\n';
|
|
533
|
+
md += '|--------|-------|\n';
|
|
534
|
+
md += `| Total Alignment Debt | ${report.summary.totalDebt.toFixed(2)} |\n`;
|
|
535
|
+
md += `| Critical Issues | ${report.summary.criticalIssues} |\n`;
|
|
536
|
+
md += `| High Priority Items | ${report.summary.highPriorityItems} |\n`;
|
|
537
|
+
md += `| Alignment Score | ${(report.summary.alignmentScore * 100).toFixed(1)}% |\n`;
|
|
538
|
+
md += `| Compliance Score | ${(report.summary.complianceScore * 100).toFixed(1)}% |\n`;
|
|
539
|
+
md += `| Drift Score | ${(report.summary.driftScore * 100).toFixed(1)}% |\n`;
|
|
540
|
+
md += '\n';
|
|
541
|
+
// Policy Violations
|
|
542
|
+
if (report.policyViolations.length > 0) {
|
|
543
|
+
md += '## Policy Violations\n\n';
|
|
544
|
+
for (const v of report.policyViolations) {
|
|
545
|
+
md += `### ${v.policyName} (${v.severity.toUpperCase()})\n\n`;
|
|
546
|
+
md += `- **Policy ID:** ${v.policyId}\n`;
|
|
547
|
+
md += `- **Description:** ${v.description}\n`;
|
|
548
|
+
if (v.location) {
|
|
549
|
+
md += `- **Location:** ${v.location}\n`;
|
|
550
|
+
}
|
|
551
|
+
if (v.suggestedFix) {
|
|
552
|
+
md += `- **Suggested Fix:** ${v.suggestedFix}\n`;
|
|
553
|
+
}
|
|
554
|
+
md += '\n';
|
|
555
|
+
}
|
|
556
|
+
}
|
|
557
|
+
// Alignment Gaps
|
|
558
|
+
if (report.alignmentGaps.length > 0) {
|
|
559
|
+
md += '## Alignment Gaps\n\n';
|
|
560
|
+
md += '| Dimension | Expected | Actual | Gap | Priority |\n';
|
|
561
|
+
md += '|-----------|----------|--------|-----|----------|\n';
|
|
562
|
+
for (const g of report.alignmentGaps) {
|
|
563
|
+
md += `| ${g.dimension} | ${(g.expected * 100).toFixed(1)}% | ${(g.actual * 100).toFixed(1)}% | ${(g.gap * 100).toFixed(1)}% | ${g.priority} |\n`;
|
|
564
|
+
}
|
|
565
|
+
md += '\n';
|
|
566
|
+
}
|
|
567
|
+
// Drift Alerts
|
|
568
|
+
if (report.driftAlerts.length > 0) {
|
|
569
|
+
md += '## Drift Alerts\n\n';
|
|
570
|
+
md += '| Pattern | Change | Direction | Severity |\n';
|
|
571
|
+
md += '|---------|--------|-----------|----------|\n';
|
|
572
|
+
for (const d of report.driftAlerts) {
|
|
573
|
+
md += `| ${d.pattern} | ${d.changePercent.toFixed(1)}% | ${d.direction} | ${d.severity} |\n`;
|
|
574
|
+
}
|
|
575
|
+
md += '\n';
|
|
576
|
+
}
|
|
577
|
+
// Recommendations
|
|
578
|
+
if (report.recommendations.length > 0) {
|
|
579
|
+
md += '## Recommendations\n\n';
|
|
580
|
+
for (const rec of report.recommendations) {
|
|
581
|
+
md += `- ${rec}\n`;
|
|
582
|
+
}
|
|
583
|
+
md += '\n';
|
|
584
|
+
}
|
|
585
|
+
return md;
|
|
586
|
+
}
|
|
587
|
+
function validateIPRESchema(config) {
|
|
588
|
+
const errors = [];
|
|
589
|
+
// Required fields
|
|
590
|
+
if (!config.version) {
|
|
591
|
+
errors.push('Missing required field: version');
|
|
592
|
+
}
|
|
593
|
+
if (!config.intent) {
|
|
594
|
+
errors.push('Missing required field: intent');
|
|
595
|
+
}
|
|
596
|
+
else {
|
|
597
|
+
if (!config.intent.mission) {
|
|
598
|
+
errors.push('Missing required field: intent.mission');
|
|
599
|
+
}
|
|
600
|
+
if (!config.intent.values || !Array.isArray(config.intent.values)) {
|
|
601
|
+
errors.push('Missing or invalid field: intent.values (must be array)');
|
|
602
|
+
}
|
|
603
|
+
}
|
|
604
|
+
if (!config.policies) {
|
|
605
|
+
errors.push('Missing required field: policies');
|
|
606
|
+
}
|
|
607
|
+
else {
|
|
608
|
+
if (!Array.isArray(config.policies.security)) {
|
|
609
|
+
errors.push('policies.security must be an array');
|
|
610
|
+
}
|
|
611
|
+
if (!Array.isArray(config.policies.compliance)) {
|
|
612
|
+
errors.push('policies.compliance must be an array');
|
|
613
|
+
}
|
|
614
|
+
if (!Array.isArray(config.policies.operational)) {
|
|
615
|
+
errors.push('policies.operational must be an array');
|
|
616
|
+
}
|
|
617
|
+
}
|
|
618
|
+
if (!config.rewards) {
|
|
619
|
+
errors.push('Missing required field: rewards');
|
|
620
|
+
}
|
|
621
|
+
else {
|
|
622
|
+
if (!config.rewards.weights) {
|
|
623
|
+
errors.push('Missing required field: rewards.weights');
|
|
624
|
+
}
|
|
625
|
+
if (typeof config.rewards.threshold !== 'number') {
|
|
626
|
+
errors.push('rewards.threshold must be a number');
|
|
627
|
+
}
|
|
628
|
+
}
|
|
629
|
+
if (!config.evaluators) {
|
|
630
|
+
errors.push('Missing required field: evaluators');
|
|
631
|
+
}
|
|
632
|
+
else if (!Array.isArray(config.evaluators)) {
|
|
633
|
+
errors.push('evaluators must be an array');
|
|
634
|
+
}
|
|
635
|
+
return errors;
|
|
636
|
+
}
|
|
637
|
+
function displayConfigSummary(config) {
|
|
638
|
+
console.log(chalk_1.default.blue('\n--- Configuration Summary ---'));
|
|
639
|
+
console.log(`Version: ${config.version}`);
|
|
640
|
+
console.log(`Mission: ${config.intent?.mission || 'Not defined'}`);
|
|
641
|
+
console.log(`Values: ${config.intent?.values?.length || 0} defined`);
|
|
642
|
+
console.log(`Security Policies: ${config.policies?.security?.length || 0}`);
|
|
643
|
+
console.log(`Compliance Policies: ${config.policies?.compliance?.length || 0}`);
|
|
644
|
+
console.log(`Operational Policies: ${config.policies?.operational?.length || 0}`);
|
|
645
|
+
console.log(`Evaluators: ${config.evaluators?.length || 0}`);
|
|
646
|
+
if (config.rewards?.threshold) {
|
|
647
|
+
console.log(`Reward Threshold: ${config.rewards.threshold}`);
|
|
648
|
+
}
|
|
649
|
+
}
|
|
650
|
+
// ============================================================================
|
|
651
|
+
// State Management
|
|
652
|
+
// ============================================================================
|
|
653
|
+
const GOVERNANCE_STATE_PATH = '.wundr/governance-state.json';
|
|
654
|
+
async function saveGovernanceState(sessionId, data) {
|
|
655
|
+
const statePath = path_1.default.join(process.cwd(), GOVERNANCE_STATE_PATH);
|
|
656
|
+
await fs_extra_1.default.ensureDir(path_1.default.dirname(statePath));
|
|
657
|
+
const state = {
|
|
658
|
+
timestamp: new Date().toISOString(),
|
|
659
|
+
sessionId,
|
|
660
|
+
suiteResult: {
|
|
661
|
+
passed: data.suiteResult.passed,
|
|
662
|
+
overallScore: data.suiteResult.overallScore,
|
|
663
|
+
criticalIssues: [...data.suiteResult.criticalIssues],
|
|
664
|
+
},
|
|
665
|
+
complianceResult: data.complianceResult,
|
|
666
|
+
alignmentResult: data.alignmentResult,
|
|
667
|
+
driftResult: data.driftResult,
|
|
668
|
+
};
|
|
669
|
+
await fs_extra_1.default.writeJson(statePath, state, { spaces: 2 });
|
|
670
|
+
}
|
|
671
|
+
async function loadGovernanceState() {
|
|
672
|
+
const statePath = path_1.default.join(process.cwd(), GOVERNANCE_STATE_PATH);
|
|
673
|
+
if (await fs_extra_1.default.pathExists(statePath)) {
|
|
674
|
+
return await fs_extra_1.default.readJson(statePath);
|
|
675
|
+
}
|
|
676
|
+
return null;
|
|
677
|
+
}
|
|
678
|
+
async function loadRecentInterventions() {
|
|
679
|
+
// Load from state file or return empty array
|
|
680
|
+
// In a real implementation, this would load from a persistent store
|
|
681
|
+
return [];
|
|
682
|
+
}
|
|
683
|
+
function extractDriftIndicatorsFromState(state) {
|
|
684
|
+
if (!state?.driftResult) {
|
|
685
|
+
return [];
|
|
686
|
+
}
|
|
687
|
+
const driftResult = state.driftResult;
|
|
688
|
+
const indicators = [];
|
|
689
|
+
for (const issue of driftResult.issues || []) {
|
|
690
|
+
const match = issue.match(/Drift.*in (\w+): ([\d.]+)%/i);
|
|
691
|
+
if (match) {
|
|
692
|
+
indicators.push({
|
|
693
|
+
pattern: match[1] ?? 'unknown',
|
|
694
|
+
baseline: 0.9,
|
|
695
|
+
current: 0.9 * (1 + parseFloat(match[2] ?? '0') / 100),
|
|
696
|
+
change: parseFloat(match[2] ?? '0') / 100,
|
|
697
|
+
severity: 'medium',
|
|
698
|
+
});
|
|
699
|
+
}
|
|
700
|
+
}
|
|
701
|
+
return indicators;
|
|
702
|
+
}
|
|
703
|
+
//# sourceMappingURL=governance.js.map
|