e5e6de2fad
- Replace bun:test with vitest across all packages - Replace bun build with esbuild - Replace bun:sqlite with better-sqlite3 - Fix OCAS Store API: store.put/get → store.cas.put/get - Fix vitest vi.mock hoisting (vi.hoisted) - Add pnpm-workspace.yaml and pnpm-lock.yaml - Update all package.json test/build scripts WIP: 8 failures remain in agent-hermes (bun engines check + sqlite migration) Refs #26
243 lines
7.3 KiB
TypeScript
243 lines
7.3 KiB
TypeScript
import { readFile } from "node:fs/promises";
|
|
import { homedir } from "node:os";
|
|
import { join } from "node:path";
|
|
|
|
import { bootstrap, type Store } from "@ocas/core";
|
|
import { createFsStore, createSqliteVarStore } from "@ocas/fs";
|
|
import type {
|
|
AgentAlias,
|
|
AgentConfig,
|
|
CasRef,
|
|
ModelAlias,
|
|
ModelConfig,
|
|
ProviderAlias,
|
|
ProviderConfig,
|
|
Scenario,
|
|
ThreadId,
|
|
ThreadIndexEntry,
|
|
WorkflowConfig,
|
|
WorkflowName,
|
|
} from "@united-workforce/protocol";
|
|
import { parse } from "yaml";
|
|
|
|
import { registerAgentSchemas } from "./schemas.js";
|
|
|
|
/** Default filesystem root for uwf data (`~/.uwf`). */
|
|
export function getDefaultStorageRoot(): string {
|
|
return join(homedir(), ".uwf");
|
|
}
|
|
|
|
/**
|
|
* Resolve storage root.
|
|
* Priority: `UWF_STORAGE_ROOT` → `WORKFLOW_STORAGE_ROOT` → default.
|
|
*/
|
|
export function resolveStorageRoot(): string {
|
|
const primary = process.env.UWF_STORAGE_ROOT;
|
|
if (primary !== undefined && primary !== "") {
|
|
return primary;
|
|
}
|
|
const userOverride = process.env.WORKFLOW_STORAGE_ROOT;
|
|
if (userOverride !== undefined && userOverride !== "") {
|
|
return userOverride;
|
|
}
|
|
return getDefaultStorageRoot();
|
|
}
|
|
|
|
export function getCasDir(storageRoot: string): string {
|
|
return join(storageRoot, "cas");
|
|
}
|
|
|
|
export function getConfigPath(storageRoot: string): string {
|
|
return join(storageRoot, "config.yaml");
|
|
}
|
|
|
|
export function getEnvPath(storageRoot: string): string {
|
|
return join(storageRoot, ".env");
|
|
}
|
|
|
|
const THREAD_VAR_PREFIX = "@uwf/thread/";
|
|
|
|
/**
|
|
* Global CAS directory (same as uwf CLI).
|
|
* Priority: `OCAS_DIR` → default ~/.ocas
|
|
*/
|
|
export function getGlobalCasDir(): string {
|
|
const primary = process.env.OCAS_DIR;
|
|
if (primary !== undefined && primary !== "") {
|
|
return primary;
|
|
}
|
|
return join(homedir(), ".ocas");
|
|
}
|
|
|
|
function threadVarName(threadId: ThreadId): string {
|
|
return `${THREAD_VAR_PREFIX}${threadId}`;
|
|
}
|
|
|
|
/** Read active thread head + suspend metadata from ocas variable store. */
|
|
export async function getActiveThreadEntry(
|
|
_storageRoot: string,
|
|
threadId: ThreadId,
|
|
): Promise<ThreadIndexEntry | null> {
|
|
const casDir = getGlobalCasDir();
|
|
const cas = createFsStore(casDir);
|
|
const { var: varStore } = createSqliteVarStore(join(casDir, "vars"), cas);
|
|
const vars = varStore.list({ exactName: threadVarName(threadId) });
|
|
const v = vars[0];
|
|
if (v === undefined) {
|
|
return null;
|
|
}
|
|
return {
|
|
head: v.value as CasRef,
|
|
suspendedRole: v.tags.suspendedRole ?? null,
|
|
suspendMessage: v.tags.suspendMessage ?? null,
|
|
};
|
|
}
|
|
|
|
export type AgentStore = {
|
|
storageRoot: string;
|
|
store: Store;
|
|
schemas: Awaited<ReturnType<typeof registerAgentSchemas>>;
|
|
};
|
|
|
|
export async function createAgentStore(storageRoot: string): Promise<AgentStore> {
|
|
const casDir = getGlobalCasDir();
|
|
const cas = createFsStore(casDir);
|
|
const { var: varSub, tag } = createSqliteVarStore(join(casDir, "vars"), cas);
|
|
const store: Store = { cas, var: varSub, tag };
|
|
bootstrap(store);
|
|
const schemas = await registerAgentSchemas(store);
|
|
return { storageRoot, store, schemas };
|
|
}
|
|
|
|
function isRecord(value: unknown): value is Record<string, unknown> {
|
|
return typeof value === "object" && value !== null && !Array.isArray(value);
|
|
}
|
|
|
|
function normalizeProviders(raw: unknown): Record<ProviderAlias, ProviderConfig> {
|
|
if (!isRecord(raw)) {
|
|
throw new Error("config.providers must be a mapping");
|
|
}
|
|
const providers: Record<ProviderAlias, ProviderConfig> = {};
|
|
for (const [name, entry] of Object.entries(raw)) {
|
|
if (!isRecord(entry)) {
|
|
throw new Error(`config.providers.${name} must be a mapping`);
|
|
}
|
|
const baseUrl = entry.baseUrl;
|
|
const apiKey = entry.apiKey;
|
|
if (typeof baseUrl !== "string" || typeof apiKey !== "string") {
|
|
throw new Error(`config.providers.${name} requires baseUrl and apiKey`);
|
|
}
|
|
providers[name] = { baseUrl, apiKey };
|
|
}
|
|
return providers;
|
|
}
|
|
|
|
function normalizeModels(raw: unknown): Record<ModelAlias, ModelConfig> {
|
|
if (!isRecord(raw)) {
|
|
throw new Error("config.models must be a mapping");
|
|
}
|
|
const models: Record<ModelAlias, ModelConfig> = {};
|
|
for (const [name, entry] of Object.entries(raw)) {
|
|
if (!isRecord(entry)) {
|
|
throw new Error(`config.models.${name} must be a mapping`);
|
|
}
|
|
const provider = entry.provider;
|
|
const modelName = entry.name;
|
|
if (typeof provider !== "string" || typeof modelName !== "string") {
|
|
throw new Error(`config.models.${name} requires provider and name`);
|
|
}
|
|
models[name] = { provider, name: modelName };
|
|
}
|
|
return models;
|
|
}
|
|
|
|
function normalizeAgents(raw: unknown): Record<AgentAlias, AgentConfig> {
|
|
if (!isRecord(raw)) {
|
|
throw new Error("config.agents must be a mapping");
|
|
}
|
|
const agents: Record<AgentAlias, AgentConfig> = {};
|
|
for (const [name, entry] of Object.entries(raw)) {
|
|
if (!isRecord(entry)) {
|
|
throw new Error(`config.agents.${name} must be a mapping`);
|
|
}
|
|
const command = entry.command;
|
|
const argsRaw = entry.args;
|
|
if (typeof command !== "string") {
|
|
throw new Error(`config.agents.${name} requires command`);
|
|
}
|
|
const args = Array.isArray(argsRaw)
|
|
? argsRaw.filter((a): a is string => typeof a === "string")
|
|
: [];
|
|
agents[name] = { command, args };
|
|
}
|
|
return agents;
|
|
}
|
|
|
|
function normalizeModelOverrides(raw: unknown): Record<Scenario, ModelAlias> | null {
|
|
if (raw === undefined || raw === null) {
|
|
return null;
|
|
}
|
|
if (!isRecord(raw)) {
|
|
throw new Error("config.modelOverrides must be a mapping or null");
|
|
}
|
|
const overrides: Record<Scenario, ModelAlias> = {};
|
|
for (const [scene, alias] of Object.entries(raw)) {
|
|
if (typeof alias === "string") {
|
|
overrides[scene] = alias;
|
|
}
|
|
}
|
|
return overrides;
|
|
}
|
|
|
|
function normalizeAgentOverrides(
|
|
raw: unknown,
|
|
): Record<WorkflowName, Record<string, AgentAlias>> | null {
|
|
if (raw === undefined || raw === null) {
|
|
return null;
|
|
}
|
|
if (!isRecord(raw)) {
|
|
throw new Error("config.agentOverrides must be a mapping or null");
|
|
}
|
|
const overrides: Record<WorkflowName, Record<string, AgentAlias>> = {};
|
|
for (const [workflowName, rolesRaw] of Object.entries(raw)) {
|
|
if (!isRecord(rolesRaw)) {
|
|
continue;
|
|
}
|
|
const roles: Record<string, AgentAlias> = {};
|
|
for (const [roleName, alias] of Object.entries(rolesRaw)) {
|
|
if (typeof alias === "string") {
|
|
roles[roleName] = alias;
|
|
}
|
|
}
|
|
overrides[workflowName] = roles;
|
|
}
|
|
return overrides;
|
|
}
|
|
|
|
export function normalizeWorkflowConfig(raw: unknown): WorkflowConfig {
|
|
if (!isRecord(raw)) {
|
|
throw new Error("config.yaml root must be a mapping");
|
|
}
|
|
const defaultAgent = raw.defaultAgent;
|
|
const defaultModel = raw.defaultModel;
|
|
if (typeof defaultAgent !== "string" || typeof defaultModel !== "string") {
|
|
throw new Error("config requires defaultAgent and defaultModel");
|
|
}
|
|
return {
|
|
providers: normalizeProviders(raw.providers),
|
|
models: normalizeModels(raw.models),
|
|
agents: normalizeAgents(raw.agents),
|
|
defaultAgent,
|
|
agentOverrides: normalizeAgentOverrides(raw.agentOverrides),
|
|
defaultModel,
|
|
modelOverrides: normalizeModelOverrides(raw.modelOverrides),
|
|
};
|
|
}
|
|
|
|
export async function loadWorkflowConfig(storageRoot: string): Promise<WorkflowConfig> {
|
|
const path = getConfigPath(storageRoot);
|
|
const text = await readFile(path, "utf8");
|
|
const raw = parse(text) as unknown;
|
|
return normalizeWorkflowConfig(raw);
|
|
}
|