feat(migration): implement workspace layout migration and testing

- Added `migrate-layout.ts` to handle migration from old workspace layouts to the new `devclaw/` structure.
- Introduced `migrate-layout.test.ts` for comprehensive tests covering various migration scenarios.
- Updated `workspace.ts` to ensure default files are created post-migration, including `workflow.yaml` and role-specific prompts.
- Refactored role instruction handling to accommodate new directory structure.
- Enhanced project registration to scaffold prompt files in the new `devclaw/projects/<project>/prompts/` directory.
- Adjusted setup tool descriptions and logic to reflect changes in file structure.
- Updated templates to align with the new workflow configuration and role instructions.
This commit is contained in:
Lauren ten Hoor
2026-02-15 20:19:09 +08:00
parent 89245f8ffa
commit a359ffed34
25 changed files with 1035 additions and 207 deletions

View File

@@ -11,6 +11,7 @@ import fs from "node:fs/promises";
import path from "node:path";
import type { OpenClawPluginApi } from "openclaw/plugin-sdk";
import { getSessionKeyRolePattern } from "./roles/index.js";
import { DATA_DIR } from "./setup/migrate-layout.js";
/**
* Parse a DevClaw subagent session key to extract project name and role.
@@ -44,18 +45,24 @@ export async function loadRoleInstructions(
projectName: string,
role: string,
): Promise<string> {
const projectFile = path.join(workspaceDir, "projects", "roles", projectName, `${role}.md`);
try {
return await fs.readFile(projectFile, "utf-8");
} catch {
/* not found — try default */
}
const defaultFile = path.join(workspaceDir, "projects", "roles", "default", `${role}.md`);
try {
return await fs.readFile(defaultFile, "utf-8");
} catch {
/* not found */
}
const dataDir = path.join(workspaceDir, DATA_DIR);
// Project-specific: devclaw/projects/<project>/prompts/<role>.md
const projectFile = path.join(dataDir, "projects", projectName, "prompts", `${role}.md`);
try { return await fs.readFile(projectFile, "utf-8"); } catch { /* not found */ }
// Fallback old path: projects/roles/<project>/<role>.md
const oldProjectFile = path.join(workspaceDir, "projects", "roles", projectName, `${role}.md`);
try { return await fs.readFile(oldProjectFile, "utf-8"); } catch { /* not found */ }
// Default: devclaw/prompts/<role>.md
const defaultFile = path.join(dataDir, "prompts", `${role}.md`);
try { return await fs.readFile(defaultFile, "utf-8"); } catch { /* not found */ }
// Fallback old default: projects/roles/default/<role>.md
const oldDefaultFile = path.join(workspaceDir, "projects", "roles", "default", `${role}.md`);
try { return await fs.readFile(oldDefaultFile, "utf-8"); } catch { /* not found */ }
return "";
}