a8b50f5109
---ci---
project: ci
phase: 6
milestone: v0.9
status: complete
artifacts:
tags: [v0.9.0]
decisions:
- id: D-047
decision: v0.9 theme = Distribution & Expansion
rationale: npm publish + OpenAI/Anthropic backends + agent flesh + parallel execution
confidence: 0.92
- id: D-049
decision: Feature milestone — patch tags v0.8.1-v0.8.6 then v0.9.0
rationale: OpenAI backend, agent flesh, npm publish all feat
confidence: 0.95
- id: D-059
decision: Rename OllamaBaseBackend to LLMBaseBackend + thin OllamaBaseBackend subclass
rationale: 15 of 17 methods backend-agnostic
confidence: 0.92
- id: D-060
decision: OpenAI/Anthropic backends use native fetch() not SDK packages
rationale: No dependency bloat; fetch native in Node 18+
confidence: 0.85
- id: D-066
decision: Concurrency limiter internal (no p-limit dependency)
rationale: 15 lines; avoids dependency for trivial feature
confidence: 0.90
- id: D-067
decision: Promise.allSettled for review agents at orchestrator lines 373-400
rationale: Current sequential loop replaced with parallel execution
confidence: 0.88
requirements:
covered: [PUBLISH-01, PUBLISH-02, PUBLISH-03, PUBLISH-04, OPENAI-01, OPENAI-02, OPENAI-03, OPENAI-04, OPENAI-05, FLESH-01, FLESH-02, FLESH-03, FLESH-04, FLESH-05, ANTHROPIC-01, ANTHROPIC-02, FLESH-06, FLESH-07, NPM-01, NPM-02, PARALLEL-01, PARALLEL-02, PARALLEL-03, INTEG-01, INTEG-02, INTEG-03, INTEG-04, INTEG-05]
---/ci---
6 phases, 28 tasks, 4077 net lines added, 57 test suites, 527 tests, zero stub agents
130 lines
4.2 KiB
TypeScript
130 lines
4.2 KiB
TypeScript
import { BackendUnavailableError, emptyTokenUsage, emptyBackendResult, DEFAULT_BACKEND_CONFIG } from "../backends/types.js";
|
|
import { OllamaLocalBackend } from "../backends/ollama-local.js";
|
|
import { OllamaCloudBackend } from "../backends/ollama-cloud.js";
|
|
import { OpencodeBackend } from "../backends/opencode.js";
|
|
|
|
describe("BackendUnavailableError", () => {
|
|
it("includes backend name in message", () => {
|
|
const err = new BackendUnavailableError("ollama-local");
|
|
expect(err.message).toContain("ollama-local");
|
|
expect(err.backendName).toBe("ollama-local");
|
|
});
|
|
|
|
it("includes agent name when provided", () => {
|
|
const err = new BackendUnavailableError("opencode", "executor");
|
|
expect(err.agentName).toBe("executor");
|
|
expect(err.message).toContain("executor");
|
|
});
|
|
});
|
|
|
|
describe("emptyTokenUsage", () => {
|
|
it("returns zeroed usage", () => {
|
|
const usage = emptyTokenUsage();
|
|
expect(usage.input_tokens).toBe(0);
|
|
expect(usage.output_tokens).toBe(0);
|
|
expect(usage.total_tokens).toBe(0);
|
|
expect(usage.estimated_cost_usd).toBe(0);
|
|
});
|
|
});
|
|
|
|
describe("emptyBackendResult", () => {
|
|
it("returns failed result with no artifacts", () => {
|
|
const result = emptyBackendResult("something failed");
|
|
expect(result.success).toBe(false);
|
|
expect(result.error).toBe("something failed");
|
|
expect(result.artifacts).toEqual([]);
|
|
expect(result.decisions).toEqual([]);
|
|
expect(result.escalations).toEqual([]);
|
|
});
|
|
|
|
it("returns result without error when no message provided", () => {
|
|
const result = emptyBackendResult();
|
|
expect(result.success).toBe(false);
|
|
expect(result.error).toBeUndefined();
|
|
});
|
|
});
|
|
|
|
describe("DEFAULT_BACKEND_CONFIG", () => {
|
|
it("has auto provider by default", () => {
|
|
expect(DEFAULT_BACKEND_CONFIG.provider).toBe("auto");
|
|
});
|
|
|
|
it("has opencode agent backend enabled", () => {
|
|
expect(DEFAULT_BACKEND_CONFIG.agent_backends.opencode?.enabled).toBe(true);
|
|
});
|
|
|
|
it("has ollama-local and ollama-cloud llm backends", () => {
|
|
expect(DEFAULT_BACKEND_CONFIG.llm_backends["openai"]).toBeDefined();
|
|
expect(DEFAULT_BACKEND_CONFIG.llm_backends["ollama-local"]).toBeDefined();
|
|
expect(DEFAULT_BACKEND_CONFIG.llm_backends["ollama-cloud"]).toBeDefined();
|
|
});
|
|
});
|
|
|
|
describe("OllamaLocalBackend", () => {
|
|
it("has correct name and type", () => {
|
|
const backend = new OllamaLocalBackend();
|
|
expect(backend.name).toBe("ollama-local");
|
|
expect(backend.type).toBe("llm");
|
|
});
|
|
|
|
it("returns false when local Ollama is not available", async () => {
|
|
const backend = new OllamaLocalBackend({
|
|
base_url: "http://localhost:1",
|
|
model_profile: "balanced",
|
|
});
|
|
const available = await backend.isAvailable();
|
|
expect(available).toBe(false);
|
|
});
|
|
|
|
it("uses default config when none provided", () => {
|
|
const backend = new OllamaLocalBackend();
|
|
expect(backend).toBeDefined();
|
|
});
|
|
});
|
|
|
|
describe("OllamaCloudBackend", () => {
|
|
it("has correct name and type", () => {
|
|
const backend = new OllamaCloudBackend();
|
|
expect(backend.name).toBe("ollama-cloud");
|
|
expect(backend.type).toBe("llm");
|
|
});
|
|
|
|
it("returns false when no base_url configured", async () => {
|
|
const backend = new OllamaCloudBackend({
|
|
base_url: "",
|
|
api_key_env: "NONEXISTENT_KEY",
|
|
model_profile: "quality",
|
|
timeout_ms: 5000,
|
|
});
|
|
const available = await backend.isAvailable();
|
|
expect(available).toBe(false);
|
|
});
|
|
|
|
it("returns false when no API key available", async () => {
|
|
const backend = new OllamaCloudBackend({
|
|
base_url: "https://example.com",
|
|
api_key_env: "NONEXISTENT_CI_KEY_12345",
|
|
model_profile: "quality",
|
|
timeout_ms: 5000,
|
|
});
|
|
const available = await backend.isAvailable();
|
|
expect(available).toBe(false);
|
|
});
|
|
});
|
|
|
|
describe("OpencodeBackend", () => {
|
|
it("has correct name and type", () => {
|
|
const backend = new OpencodeBackend();
|
|
expect(backend.name).toBe("opencode");
|
|
expect(backend.type).toBe("agent");
|
|
});
|
|
|
|
it("returns false when opencode is not installed", async () => {
|
|
const backend = new OpencodeBackend({
|
|
enabled: true,
|
|
executable: "nonexistent-opencode-binary-xyz",
|
|
});
|
|
const available = await backend.isAvailable();
|
|
expect(available).toBe(false);
|
|
});
|
|
}); |