4a58aa1657
- Type renames: CIConfig → CIAgentConfig, DEFAULT_CI_CONFIG → DEFAULT_CIAGENT_CONFIG - Type renames: CiMetadata → CIAgentMetadata, ParsedCiCommit → ParsedCIAgentCommit - Function renames: initCI → initCIAgent, isCIInitialized → isCIAgentInitialized - Function renames: extractCiBlock → extractCIAgentBlock, parseCiBlock → parseCIAgentBlock - Class renames: CiFiles → CIAgentFiles - Import paths: ci-files.js → ciagent-files.js - Directory paths: .ci/ → .ciagent/ across all source and test files - Check names: ".ci directory exists" → ".ciagent directory exists" - Check names: "CI config valid" → "CIAgent config valid" - Temp dir names: ci-*-test- → ciagent-*-test- - CLI examples: "ci init" → "ciagent init" - Fix deepMerge infinite recursion bug in config.ts - ---ci---/---/ci--- block markers preserved unchanged - All 31 test suites, 370 tests passing ---ci--- phase: 1 milestone: v0.5 plan: 07 task: 07-01-01 status: execute ---/ci---
134 lines
3.4 KiB
TypeScript
134 lines
3.4 KiB
TypeScript
import * as fs from "node:fs";
|
|
import * as path from "node:path";
|
|
import { Decision } from "../types/decisions.js";
|
|
import { Escalation } from "../types/escalation.js";
|
|
|
|
export interface AuditEntry {
|
|
phase: number;
|
|
decisions: Decision[];
|
|
escalations: Escalation[];
|
|
}
|
|
|
|
const AUDIT_DIR = "audit";
|
|
|
|
function getAuditDir(projectPath: string): string {
|
|
return path.join(projectPath, ".ciagent", AUDIT_DIR);
|
|
}
|
|
|
|
function getAuditFilePath(projectPath: string, phase: number): string {
|
|
const date = new Date().toISOString().split("T")[0];
|
|
return path.join(getAuditDir(projectPath), `${date}-phase${phase}-decisions.json`);
|
|
}
|
|
|
|
function ensureAuditDir(projectPath: string): void {
|
|
const dir = getAuditDir(projectPath);
|
|
if (!fs.existsSync(dir)) {
|
|
fs.mkdirSync(dir, { recursive: true });
|
|
}
|
|
}
|
|
|
|
export function logDecision(
|
|
projectPath: string,
|
|
phase: number,
|
|
decision: Decision
|
|
): void {
|
|
ensureAuditDir(projectPath);
|
|
const filePath = getAuditFilePath(projectPath, phase);
|
|
let entry: AuditEntry;
|
|
|
|
if (fs.existsSync(filePath)) {
|
|
entry = JSON.parse(fs.readFileSync(filePath, "utf-8"));
|
|
} else {
|
|
entry = { phase, decisions: [], escalations: [] };
|
|
}
|
|
|
|
entry.decisions.push(decision);
|
|
fs.writeFileSync(filePath, JSON.stringify(entry, null, 2), "utf-8");
|
|
}
|
|
|
|
export function logEscalation(
|
|
projectPath: string,
|
|
phase: number,
|
|
escalation: Escalation
|
|
): void {
|
|
ensureAuditDir(projectPath);
|
|
const filePath = getAuditFilePath(projectPath, phase);
|
|
let entry: AuditEntry;
|
|
|
|
if (fs.existsSync(filePath)) {
|
|
entry = JSON.parse(fs.readFileSync(filePath, "utf-8"));
|
|
} else {
|
|
entry = { phase, decisions: [], escalations: [] };
|
|
}
|
|
|
|
entry.escalations.push(escalation);
|
|
fs.writeFileSync(filePath, JSON.stringify(entry, null, 2), "utf-8");
|
|
}
|
|
|
|
export function readAudit(
|
|
projectPath: string,
|
|
phase?: number
|
|
): AuditEntry[] {
|
|
const auditDir = getAuditDir(projectPath);
|
|
if (!fs.existsSync(auditDir)) return [];
|
|
|
|
const files = fs
|
|
.readdirSync(auditDir)
|
|
.filter((f) => f.endsWith("-decisions.json"))
|
|
.sort();
|
|
|
|
const entries: AuditEntry[] = [];
|
|
for (const file of files) {
|
|
const content = fs.readFileSync(path.join(auditDir, file), "utf-8");
|
|
const entry: AuditEntry = JSON.parse(content);
|
|
if (phase === undefined || entry.phase === phase) {
|
|
entries.push(entry);
|
|
}
|
|
}
|
|
|
|
return entries;
|
|
}
|
|
|
|
export function getAuditSummary(projectPath: string): {
|
|
total_decisions: number;
|
|
total_escalations: number;
|
|
phases: number[];
|
|
decisions_by_confidence: Record<string, number>;
|
|
escalations_by_type: Record<string, number>;
|
|
} {
|
|
const entries = readAudit(projectPath);
|
|
let total_decisions = 0;
|
|
let total_escalations = 0;
|
|
const phases = new Set<number>();
|
|
const decisions_by_confidence: Record<string, number> = {
|
|
high: 0,
|
|
medium: 0,
|
|
low: 0,
|
|
};
|
|
const escalations_by_type: Record<string, number> = {};
|
|
|
|
for (const entry of entries) {
|
|
phases.add(entry.phase);
|
|
total_decisions += entry.decisions.length;
|
|
total_escalations += entry.escalations.length;
|
|
|
|
for (const d of entry.decisions) {
|
|
const level =
|
|
d.confidence > 0.85 ? "high" : d.confidence >= 0.6 ? "medium" : "low";
|
|
decisions_by_confidence[level]++;
|
|
}
|
|
|
|
for (const e of entry.escalations) {
|
|
escalations_by_type[e.type] =
|
|
(escalations_by_type[e.type] || 0) + 1;
|
|
}
|
|
}
|
|
|
|
return {
|
|
total_decisions,
|
|
total_escalations,
|
|
phases: [...phases],
|
|
decisions_by_confidence,
|
|
escalations_by_type,
|
|
};
|
|
} |