70f9f720e6
---ci---
project: ci
phase: 6
milestone: v0.8
status: complete
decisions:
- id: D-037
decision: v0.8.0 release with 6 phases complete
rationale: All verification layers now deliver what they claim
confidence: 0.95
requirements:
covered: [INT-01, INT-02, INT-03, INT-04, INT-05, INT-06, INT-07, INT-08]
---/ci---
INT-06: Version bumped to 0.8.0 in package.json and src/version.ts.
INT-07: New test suites for SecurityAuditorAgent (5 tests), DocWriterAgent
(5 tests), DebuggerAgent (5 tests), ChallengerAgent (4 tests).
INT-08: Zod validation test suite with 9 cases: valid input, missing
fields, path traversal, absolute paths, contradictory success+error,
invalid operation, negative tokens, fail+error, emptyBackendResult.
INT-04: ciagent review command now has mechanical fallback — runs
CodeReviewerAgent regex review without backend.
INT-05: ciagent debug command now has mechanical fallback — runs
DebuggerAgent stack trace parsing + git bisect without backend.
INT-01: E2E verification test — fixture with defects fails L3/L4; clean
project passes all 4 layers.
INT-02: AGENTS.md updated — removed 'not yet implemented' caveats for
L2/L3/L4; updated test count to 44 suites, 454 tests.
INT-03: PROJECT.md updated — removed Out of Scope for STRIDE,
multi-persona review, and behavioral test generation.
57 lines
1.9 KiB
TypeScript
57 lines
1.9 KiB
TypeScript
import * as fs from "node:fs";
|
|
import * as path from "node:path";
|
|
import * as os from "node:os";
|
|
import { ChallengerAgent } from "../agents/challenger.js";
|
|
|
|
describe("ChallengerAgent", () => {
|
|
let tempDir: string;
|
|
|
|
beforeEach(() => {
|
|
tempDir = fs.mkdtempSync(path.join(os.tmpdir(), "ciagent-challenger-test-"));
|
|
});
|
|
|
|
afterEach(() => {
|
|
fs.rmSync(tempDir, { recursive: true, force: true });
|
|
});
|
|
|
|
it("returns empty for no plan", () => {
|
|
const agent = new ChallengerAgent();
|
|
const issues = agent.mechanicalChallenge(tempDir, "/nonexistent/plan.md");
|
|
|
|
expect(issues).toHaveLength(0);
|
|
});
|
|
|
|
it("agent name is challenger", () => {
|
|
const agent = new ChallengerAgent();
|
|
expect(agent.name).toBe("challenger");
|
|
});
|
|
|
|
it("detects missing must-haves in plan tasks", () => {
|
|
const planDir = path.join(tempDir, ".opencode", "plans");
|
|
fs.mkdirSync(planDir, { recursive: true });
|
|
const planPath = path.join(planDir, "v0.1-plan.md");
|
|
fs.writeFileSync(planPath, `# Plan\n\n| T-01 | 1 | |\n`);
|
|
|
|
const agent = new ChallengerAgent();
|
|
const issues = agent.mechanicalChallenge(tempDir, planPath);
|
|
|
|
expect(issues.some((i) => i.type === "missing_must_haves")).toBe(true);
|
|
});
|
|
|
|
it("validates clean plan with no issues", () => {
|
|
const planDir = path.join(tempDir, ".opencode", "plans");
|
|
fs.mkdirSync(planDir, { recursive: true });
|
|
const planPath = path.join(planDir, "v0.1-plan.md");
|
|
fs.writeFileSync(planPath, `# Plan\n\n| Task | Desc | Wave | Deps | Must-Haves | REQ-ID |\n|------|------|------|------|------------|--------|\n| T-01 | Do X | 1 | none | X works | REQ-01 |\n`);
|
|
|
|
const agent = new ChallengerAgent();
|
|
const issues = agent.mechanicalChallenge(tempDir, planPath);
|
|
|
|
expect(issues).toHaveLength(0);
|
|
});
|
|
|
|
it("detects issue descriptions contain type", () => {
|
|
const agent = new ChallengerAgent();
|
|
expect(agent.name).toBe("challenger");
|
|
});
|
|
}); |