* Workflow commands for orchestrating multi-agent workflows.
*
* Commands are embedded at build time via Bun's import with { type: "text" }.
*/
import * as path from "node:path";
import { parseFrontmatter, prompt } from "@oh-my-pi/pi-utils";
import { type SlashCommand, slashCommandCapability } from "../capability/slash-command";
import { loadCapability } from "../discovery";
import initMd from "../prompts/agents/init.md" with { type: "text" };
const EMBEDDED_COMMANDS: { name: string; content: string }[] = [{ name: "init.md", content: prompt.render(initMd) }];
export const EMBEDDED_COMMAND_TEMPLATES: ReadonlyArray<{ name: string; content: string }> = EMBEDDED_COMMANDS;
export interface WorkflowCommand {
name: string;
description: string;
instructions: string;
source: "bundled" | "user" | "project";
filePath: string;
}
function getString(frontmatter: Record<string, unknown>, key: string): string {
const value = frontmatter[key];
return typeof value === "string" ? value : "";
}
let bundledCommandsCache: WorkflowCommand[] | null = null;
* Load all bundled commands from embedded content.
*/
export function loadBundledCommands(): WorkflowCommand[] {
if (bundledCommandsCache !== null) {
return bundledCommandsCache;
}
const commands: WorkflowCommand[] = [];
for (const { name, content } of EMBEDDED_COMMANDS) {
const { frontmatter, body } = parseFrontmatter(content, {
source: `embedded:${name}`,
level: "fatal",
});
const cmdName = name.replace(/\.md$/, "");
commands.push({
name: cmdName,
description: getString(frontmatter, "description"),
instructions: body,
source: "bundled",
filePath: `embedded:${name}`,
});
}
bundledCommandsCache = commands;
return commands;
}
* Discover all available commands.
*
* Precedence (highest wins): .omp > .pi > .claude (project before user), then bundled
*/
export async function discoverCommands(cwd: string): Promise<WorkflowCommand[]> {
const resolvedCwd = path.resolve(cwd);
const result = await loadCapability<SlashCommand>(slashCommandCapability.id, { cwd: resolvedCwd });
const commands: WorkflowCommand[] = [];
const seen = new Set<string>();
for (const cmd of result.items) {
if (seen.has(cmd.name)) continue;
const { frontmatter, body } = parseFrontmatter(cmd.content, {
source: cmd.path ?? `workflow-command:${cmd.name}`,
level: cmd.level === "native" ? "fatal" : "warn",
});
const source: "bundled" | "user" | "project" = cmd.level === "native" ? "bundled" : cmd.level;
commands.push({
name: cmd.name,
description: getString(frontmatter, "description"),
instructions: body,
source,
filePath: cmd.path,
});
seen.add(cmd.name);
}
for (const cmd of loadBundledCommands()) {
if (seen.has(cmd.name)) continue;
commands.push(cmd);
seen.add(cmd.name);
}
return commands;
}
* Get a command by name.
*/
export function getCommand(commands: WorkflowCommand[], name: string): WorkflowCommand | undefined {
return commands.find(c => c.name === name);
}
* Expand command instructions with task input.
* Replaces $@ with the provided input.
*/
export function expandCommand(command: WorkflowCommand, input: string): string {
return command.instructions.replace(/\$@/g, input);
}
* Clear the bundled commands cache (for testing).
*/
export function clearBundledCommandsCache(): void {
bundledCommandsCache = null;
}