68079cc003
CI / check (pull_request) Successful in 1m43s
- Validator: hasStatusConst/getConstStatuses replace enum checks - enum in $status is now rejected with clear error message - All docs/examples/tests migrated from enum to const/oneOf - bootstrap hello.yaml updated Fixes #123
199 lines
5.0 KiB
TypeScript
199 lines
5.0 KiB
TypeScript
import { mkdir, rm, writeFile } from "node:fs/promises";
|
|
import { tmpdir } from "node:os";
|
|
import { join } from "node:path";
|
|
import type { CasRef, StartNodePayload, ThreadId } from "@united-workforce/protocol";
|
|
import { describe, expect, test } from "vitest";
|
|
import { cmdThreadStart } from "../commands/thread.js";
|
|
import { createUwfStore, getThread } from "../store.js";
|
|
|
|
describe("Thread and edge location integration", () => {
|
|
let tmpDir: string;
|
|
let storageRoot: string;
|
|
let casDir: string;
|
|
let originalEnv: string | undefined;
|
|
|
|
async function setupTestEnv() {
|
|
tmpDir = join(tmpdir(), `uwf-test-location-${Date.now()}`);
|
|
storageRoot = join(tmpDir, "storage");
|
|
casDir = join(tmpDir, "cas");
|
|
await mkdir(storageRoot, { recursive: true });
|
|
await mkdir(casDir, { recursive: true });
|
|
|
|
// Set OCAS_HOME for this test
|
|
originalEnv = process.env.OCAS_HOME;
|
|
process.env.OCAS_HOME = casDir;
|
|
}
|
|
|
|
async function teardown() {
|
|
if (tmpDir) {
|
|
await rm(tmpDir, { recursive: true, force: true });
|
|
}
|
|
// Restore original environment
|
|
if (originalEnv === undefined) {
|
|
delete process.env.OCAS_HOME;
|
|
} else {
|
|
process.env.OCAS_HOME = originalEnv;
|
|
}
|
|
}
|
|
|
|
test("thread start captures cwd in StartNode", async () => {
|
|
await setupTestEnv();
|
|
|
|
const workflowYaml = `
|
|
name: test-location
|
|
description: Test workflow for location feature
|
|
roles:
|
|
planner:
|
|
description: Plans the work
|
|
goal: Plan implementation
|
|
capabilities: ["planning"]
|
|
procedure: Plan
|
|
output: |
|
|
$status: "ready"
|
|
frontmatter:
|
|
type: object
|
|
required: ["$status"]
|
|
properties:
|
|
$status: { const: "ready" }
|
|
graph:
|
|
$START:
|
|
new:
|
|
role: planner
|
|
prompt: "Plan the work"
|
|
location: null
|
|
resume:
|
|
role: planner
|
|
prompt: "Resume the work"
|
|
location: null
|
|
planner:
|
|
ready:
|
|
role: $END
|
|
prompt: "Done"
|
|
location: null
|
|
`;
|
|
|
|
const workflowPath = join(tmpDir, "test-location.yaml");
|
|
await writeFile(workflowPath, workflowYaml, "utf8");
|
|
|
|
const testCwd = "/test/project/path";
|
|
const result = await cmdThreadStart(storageRoot, workflowPath, "test prompt", tmpDir, testCwd);
|
|
|
|
expect(result.thread).toBeDefined();
|
|
expect(result.workflow).toBeDefined();
|
|
|
|
// Verify StartNode has the cwd field
|
|
const uwf = await createUwfStore(storageRoot);
|
|
const headHash = getThread(uwf.varStore, result.thread as ThreadId)!.head;
|
|
expect(headHash).toBeDefined();
|
|
|
|
const startNode = uwf.store.cas.get(headHash as CasRef);
|
|
expect(startNode).not.toBe(null);
|
|
expect(startNode?.type).toBe(uwf.schemas.startNode);
|
|
|
|
const startPayload = startNode?.payload as StartNodePayload;
|
|
expect(startPayload.cwd).toBe(testCwd);
|
|
|
|
await teardown();
|
|
});
|
|
|
|
test("thread start validates cwd is absolute path", async () => {
|
|
await setupTestEnv();
|
|
|
|
const workflowYaml = `
|
|
name: test-location
|
|
description: Test workflow
|
|
roles:
|
|
planner:
|
|
description: Plans
|
|
goal: Plan
|
|
capabilities: ["planning"]
|
|
procedure: Plan
|
|
output: |
|
|
$status: "ready"
|
|
frontmatter:
|
|
type: object
|
|
required: ["$status"]
|
|
properties:
|
|
$status: { const: "ready" }
|
|
graph:
|
|
$START:
|
|
new:
|
|
role: planner
|
|
prompt: "Plan"
|
|
location: null
|
|
resume:
|
|
role: planner
|
|
prompt: "Resume"
|
|
location: null
|
|
planner:
|
|
ready:
|
|
role: $END
|
|
prompt: "Done"
|
|
location: null
|
|
`;
|
|
|
|
const workflowPath = join(tmpDir, "test-location.yaml");
|
|
await writeFile(workflowPath, workflowYaml, "utf8");
|
|
|
|
// Relative path should fail via fail() → process.exit (mocked in test preload)
|
|
await expect(
|
|
cmdThreadStart(storageRoot, workflowPath, "test", tmpDir, "relative/path"),
|
|
).rejects.toThrow();
|
|
|
|
await teardown();
|
|
});
|
|
|
|
test("thread start uses process.cwd() as default", async () => {
|
|
await setupTestEnv();
|
|
|
|
const workflowYaml = `
|
|
name: test-default-cwd
|
|
description: Test default cwd
|
|
roles:
|
|
planner:
|
|
description: Plans
|
|
goal: Plan
|
|
capabilities: ["planning"]
|
|
procedure: Plan
|
|
output: |
|
|
$status: "ready"
|
|
frontmatter:
|
|
type: object
|
|
required: ["$status"]
|
|
properties:
|
|
$status: { const: "ready" }
|
|
graph:
|
|
$START:
|
|
new:
|
|
role: planner
|
|
prompt: "Plan"
|
|
location: null
|
|
resume:
|
|
role: planner
|
|
prompt: "Resume"
|
|
location: null
|
|
planner:
|
|
ready:
|
|
role: $END
|
|
prompt: "Done"
|
|
location: null
|
|
`;
|
|
|
|
const workflowPath = join(tmpDir, "test-default-cwd.yaml");
|
|
await writeFile(workflowPath, workflowYaml, "utf8");
|
|
|
|
const result = await cmdThreadStart(storageRoot, workflowPath, "test", tmpDir);
|
|
|
|
const uwf = await createUwfStore(storageRoot);
|
|
const headHash = getThread(uwf.varStore, result.thread as ThreadId)!.head;
|
|
|
|
const startNode = uwf.store.cas.get(headHash as CasRef);
|
|
const startPayload = startNode?.payload as StartNodePayload;
|
|
|
|
// Should default to process.cwd()
|
|
expect(startPayload.cwd).toBe(process.cwd());
|
|
|
|
await teardown();
|
|
});
|
|
});
|