- Removed the deprecated tiers.ts file and migrated all related functionality to roles/index.js. - Updated tests and tools to reflect the new role structure, replacing references to "dev", "qa", and "architect" with "developer", "tester", and "architect". - Adjusted workflow configurations and state management to accommodate the new role naming conventions. - Enhanced project registration and health check tools to support dynamic role handling. - Updated task creation, update, and completion processes to align with the new role definitions. - Improved documentation and comments to clarify role responsibilities and usage.
204 lines
7.4 KiB
TypeScript
204 lines
7.4 KiB
TypeScript
/**
|
|
* roles/selectors.ts — Query helpers for the role registry.
|
|
*
|
|
* All role-related lookups go through these functions.
|
|
* No other file should access ROLE_REGISTRY directly for role logic.
|
|
*/
|
|
import { ROLE_REGISTRY } from "./registry.js";
|
|
import type { RoleConfig } from "./types.js";
|
|
import type { ResolvedRoleConfig } from "../config/types.js";
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Role IDs
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** All registered role IDs. */
|
|
export function getAllRoleIds(): string[] {
|
|
return Object.keys(ROLE_REGISTRY);
|
|
}
|
|
|
|
/** The role ID union type, derived from registry. */
|
|
export type WorkerRole = keyof typeof ROLE_REGISTRY;
|
|
|
|
/** Check if a string is a valid role ID. */
|
|
export function isValidRole(role: string): boolean {
|
|
return role in ROLE_REGISTRY;
|
|
}
|
|
|
|
/** Get role config by ID. Returns undefined if not found. */
|
|
export function getRole(role: string): RoleConfig | undefined {
|
|
return ROLE_REGISTRY[role];
|
|
}
|
|
|
|
/** Get role config by ID. Throws if not found. */
|
|
export function requireRole(role: string): RoleConfig {
|
|
const config = ROLE_REGISTRY[role];
|
|
if (!config) throw new Error(`Unknown role: "${role}". Valid roles: ${getAllRoleIds().join(", ")}`);
|
|
return config;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Role aliases — maps old role IDs to new canonical IDs
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Maps old role IDs to canonical IDs. Used for backward compatibility. */
|
|
export const ROLE_ALIASES: Record<string, string> = {
|
|
dev: "developer",
|
|
qa: "tester",
|
|
};
|
|
|
|
/** Resolve a role ID, applying aliases for backward compatibility. */
|
|
export function canonicalRole(role: string): string {
|
|
return ROLE_ALIASES[role] ?? role;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Level aliases — maps old level names to new canonical names
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Maps old level names to canonical names, per role. Used for backward compatibility. */
|
|
export const LEVEL_ALIASES: Record<string, Record<string, string>> = {
|
|
developer: { mid: "medior", medior: "medior" },
|
|
dev: { mid: "medior", medior: "medior" },
|
|
tester: { mid: "medior", reviewer: "medior", tester: "junior" },
|
|
qa: { mid: "medior", reviewer: "medior", tester: "junior" },
|
|
architect: { opus: "senior", sonnet: "junior" },
|
|
};
|
|
|
|
/** Resolve a level name, applying aliases for backward compatibility. */
|
|
export function canonicalLevel(role: string, level: string): string {
|
|
return LEVEL_ALIASES[role]?.[level] ?? level;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Levels
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Get valid levels for a role. */
|
|
export function getLevelsForRole(role: string): readonly string[] {
|
|
return getRole(role)?.levels ?? [];
|
|
}
|
|
|
|
/** Get all levels across all roles. */
|
|
export function getAllLevels(): string[] {
|
|
return Object.values(ROLE_REGISTRY).flatMap(r => [...r.levels]);
|
|
}
|
|
|
|
/** Check if a level belongs to a specific role. */
|
|
export function isLevelForRole(level: string, role: string): boolean {
|
|
return getLevelsForRole(role).includes(level);
|
|
}
|
|
|
|
/** Determine which role a level belongs to. Returns undefined if no match. */
|
|
export function roleForLevel(level: string): string | undefined {
|
|
for (const [roleId, config] of Object.entries(ROLE_REGISTRY)) {
|
|
if (config.levels.includes(level)) return roleId;
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
/** Get the default level for a role. */
|
|
export function getDefaultLevel(role: string): string | undefined {
|
|
return getRole(role)?.defaultLevel;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Models
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Get default model for a role + level. */
|
|
export function getDefaultModel(role: string, level: string): string | undefined {
|
|
return getRole(role)?.models[level];
|
|
}
|
|
|
|
/** Get all default models, nested by role (for config schema). */
|
|
export function getAllDefaultModels(): Record<string, Record<string, string>> {
|
|
const result: Record<string, Record<string, string>> = {};
|
|
for (const [roleId, config] of Object.entries(ROLE_REGISTRY)) {
|
|
result[roleId] = { ...config.models };
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Resolve a level to a full model ID.
|
|
*
|
|
* Resolution order:
|
|
* 1. Plugin config `models.<role>.<level>` in openclaw.json (highest precedence)
|
|
* 2. Resolved config from config.yaml (if provided)
|
|
* 3. Registry default model
|
|
* 4. Passthrough (treat level as raw model ID)
|
|
*/
|
|
export function resolveModel(
|
|
role: string,
|
|
level: string,
|
|
pluginConfig?: Record<string, unknown>,
|
|
resolvedRole?: ResolvedRoleConfig,
|
|
): string {
|
|
const canonical = canonicalLevel(role, level);
|
|
|
|
// 1. Plugin config override (openclaw.json) — check canonical role + old aliases
|
|
const models = (pluginConfig as { models?: Record<string, unknown> })?.models;
|
|
if (models && typeof models === "object") {
|
|
// Check canonical role name, then fall back to old aliases (e.g., "dev" for "developer")
|
|
const roleModels = (models[role] ?? models[Object.entries(ROLE_ALIASES).find(([, v]) => v === role)?.[0] ?? ""]) as Record<string, string> | undefined;
|
|
if (roleModels?.[canonical]) return roleModels[canonical];
|
|
if (roleModels?.[level]) return roleModels[level];
|
|
}
|
|
|
|
// 2. Resolved config (config.yaml)
|
|
if (resolvedRole?.models[canonical]) return resolvedRole.models[canonical];
|
|
|
|
// 3. Built-in registry default
|
|
return getDefaultModel(role, canonical) ?? canonical;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Emoji
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Get emoji for a role + level. */
|
|
export function getEmoji(role: string, level: string): string | undefined {
|
|
return getRole(role)?.emoji[level];
|
|
}
|
|
|
|
/** Get fallback emoji for a role. */
|
|
export function getFallbackEmoji(role: string): string {
|
|
return getRole(role)?.fallbackEmoji ?? "📋";
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Completion
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Get valid completion results for a role. */
|
|
export function getCompletionResults(role: string): readonly string[] {
|
|
return getRole(role)?.completionResults ?? [];
|
|
}
|
|
|
|
/** Check if a result is valid for a role. */
|
|
export function isValidResult(role: string, result: string): boolean {
|
|
return getCompletionResults(role).includes(result);
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Session keys
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Build regex pattern that matches any registered role in session keys. */
|
|
export function getSessionKeyRolePattern(): string {
|
|
return Object.values(ROLE_REGISTRY).map(r => r.sessionKeyPattern).join("|");
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Notifications
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/** Check if a role has a specific notification enabled. */
|
|
export function isNotificationEnabled(
|
|
role: string,
|
|
event: "onStart" | "onComplete",
|
|
): boolean {
|
|
return getRole(role)?.notifications[event] ?? true;
|
|
}
|