Every role is self-contained (types.ts, prompt.ts, index.ts).
No shared.ts, no cross-role imports. All dependencies injected:
index.ts — wiring (resolve env, call buildSenseGenerator)
build.ts — buildSenseGenerator(deps) → WorkflowDefinition
moderator.ts — pure routing, composes meta from role types
roles/planner/ — buildPlannerRole(deps), self-contained
roles/coder/ — buildCoderRole(deps), self-contained
roles/tester/ — buildTesterRole(deps), self-contained
Workflow is now reusable: buildSenseGenerator() can be called with
any provider/paths, not hardcoded to this machine.
小橘 🍊(NEKO Team)
28 lines
919 B
TypeScript
28 lines
919 B
TypeScript
import { END } from "@uncaged/nerve-core";
|
|
import type { Moderator } from "@uncaged/nerve-core";
|
|
import type { PlannerMeta } from "./roles/planner/types.js";
|
|
import type { CoderMeta } from "./roles/coder/types.js";
|
|
import type { TesterMeta } from "./roles/tester/types.js";
|
|
|
|
export type SenseMeta = {
|
|
planner: PlannerMeta;
|
|
coder: CoderMeta;
|
|
tester: TesterMeta;
|
|
};
|
|
|
|
function countRole(steps: { role: string }[], name: string): number {
|
|
return steps.filter((s) => s.role === name).length;
|
|
}
|
|
|
|
export const moderator: Moderator<SenseMeta> = (context) => {
|
|
if (context.steps.length === 0) return "planner";
|
|
const last = context.steps[context.steps.length - 1];
|
|
if (last.role === "planner") return "coder";
|
|
if (last.role === "coder") return "tester";
|
|
if (last.role === "tester") {
|
|
if (last.meta.passed) return END;
|
|
return countRole(context.steps, "tester") < 3 ? "coder" : END;
|
|
}
|
|
return END;
|
|
};
|