- Moved setup logic into dedicated files: agent.ts, config.ts, index.ts, workspace.ts. - Introduced tool-helpers.ts for shared functions across tools, reducing boilerplate. - Updated tools (status, task-comment, task-create, task-update, work-finish, work-start) to utilize new helper functions for workspace resolution and provider creation. - Enhanced error handling and context detection in tools. - Improved project resolution logic to streamline tool execution. - Added new functionality for agent creation and configuration management in setup.
134 lines
6.8 KiB
TypeScript
134 lines
6.8 KiB
TypeScript
/**
|
|
* work_start — Pick up a task from the issue queue.
|
|
*
|
|
* Context-aware: ONLY works in project group chats.
|
|
* Auto-detects: projectGroupId, role, tier, issueId.
|
|
* After dispatch, ticks the project queue to fill parallel slots.
|
|
*/
|
|
import type { OpenClawPluginApi } from "openclaw/plugin-sdk";
|
|
import { jsonResult } from "openclaw/plugin-sdk";
|
|
import type { ToolContext } from "../types.js";
|
|
import type { StateLabel } from "../providers/provider.js";
|
|
import { selectTier } from "../model-selector.js";
|
|
import { getWorker } from "../projects.js";
|
|
import { dispatchTask } from "../dispatch.js";
|
|
import { notify, getNotificationConfig } from "../notify.js";
|
|
import { findNextIssue, detectRoleFromLabel, detectTierFromLabels, projectTick, type TickResult } from "../services/tick.js";
|
|
import { requireWorkspaceDir, resolveContext, resolveProject, resolveProvider, groupOnlyError, getPluginConfig } from "../tool-helpers.js";
|
|
|
|
export function createWorkStartTool(api: OpenClawPluginApi) {
|
|
return (ctx: ToolContext) => ({
|
|
name: "work_start",
|
|
label: "Work Start",
|
|
description: `Pick up a task from the issue queue. ONLY works in project group chats. Handles label transition, tier assignment, session creation, dispatch, audit, and ticks the queue to fill parallel slots.`,
|
|
parameters: {
|
|
type: "object",
|
|
properties: {
|
|
issueId: { type: "number", description: "Issue ID. If omitted, picks next by priority." },
|
|
role: { type: "string", enum: ["dev", "qa"], description: "Worker role. Auto-detected from label if omitted." },
|
|
projectGroupId: { type: "string", description: "Project group ID. Auto-detected from group context." },
|
|
tier: { type: "string", description: "Developer tier (junior/medior/senior/qa). Auto-detected if omitted." },
|
|
},
|
|
},
|
|
|
|
async execute(_id: string, params: Record<string, unknown>) {
|
|
const issueIdParam = params.issueId as number | undefined;
|
|
const roleParam = params.role as "dev" | "qa" | undefined;
|
|
const groupIdParam = params.projectGroupId as string | undefined;
|
|
const tierParam = params.tier as string | undefined;
|
|
const workspaceDir = requireWorkspaceDir(ctx);
|
|
|
|
// Context guard: group only
|
|
const context = await resolveContext(ctx, api);
|
|
if (context.type !== "group") return groupOnlyError("work_start", context);
|
|
|
|
const groupId = groupIdParam ?? context.groupId;
|
|
const { project } = await resolveProject(workspaceDir, groupId);
|
|
const { provider } = resolveProvider(project);
|
|
|
|
// Find issue
|
|
let issue: { iid: number; title: string; description: string; labels: string[]; web_url: string; state: string };
|
|
let currentLabel: StateLabel;
|
|
if (issueIdParam !== undefined) {
|
|
issue = await provider.getIssue(issueIdParam);
|
|
const label = provider.getCurrentStateLabel(issue);
|
|
if (!label) throw new Error(`Issue #${issueIdParam} has no recognized state label`);
|
|
currentLabel = label;
|
|
} else {
|
|
const next = await findNextIssue(provider, roleParam);
|
|
if (!next) return jsonResult({ success: false, error: `No issues available. Queue is empty.` });
|
|
issue = next.issue;
|
|
currentLabel = next.label;
|
|
}
|
|
|
|
// Detect role
|
|
const detectedRole = detectRoleFromLabel(currentLabel);
|
|
if (!detectedRole) throw new Error(`Label "${currentLabel}" doesn't map to a role`);
|
|
const role = roleParam ?? detectedRole;
|
|
if (roleParam && roleParam !== detectedRole) throw new Error(`Role mismatch: "${currentLabel}" → ${detectedRole}, requested ${roleParam}`);
|
|
|
|
// Check worker availability
|
|
const worker = getWorker(project, role);
|
|
if (worker.active) throw new Error(`${role.toUpperCase()} already active on ${project.name} (issue: ${worker.issueId})`);
|
|
if ((project.roleExecution ?? "parallel") === "sequential") {
|
|
const other = role === "dev" ? "qa" : "dev";
|
|
if (getWorker(project, other).active) throw new Error(`Sequential roleExecution: ${other.toUpperCase()} is active`);
|
|
}
|
|
|
|
// Select tier
|
|
const targetLabel: StateLabel = role === "dev" ? "Doing" : "Testing";
|
|
let selectedTier: string, tierReason: string, tierSource: string;
|
|
if (tierParam) {
|
|
selectedTier = tierParam; tierReason = "LLM-selected"; tierSource = "llm";
|
|
} else {
|
|
const labelTier = detectTierFromLabels(issue.labels);
|
|
if (labelTier) {
|
|
if (role === "qa" && labelTier !== "qa") { selectedTier = "qa"; tierReason = `QA overrides "${labelTier}"`; tierSource = "role-override"; }
|
|
else if (role === "dev" && labelTier === "qa") { const s = selectTier(issue.title, issue.description ?? "", role); selectedTier = s.tier; tierReason = s.reason; tierSource = "heuristic"; }
|
|
else { selectedTier = labelTier; tierReason = `Label: "${labelTier}"`; tierSource = "label"; }
|
|
} else {
|
|
const s = selectTier(issue.title, issue.description ?? "", role);
|
|
selectedTier = s.tier; tierReason = s.reason; tierSource = "heuristic";
|
|
}
|
|
}
|
|
|
|
// Dispatch
|
|
const pluginConfig = getPluginConfig(api);
|
|
const dr = await dispatchTask({
|
|
workspaceDir, agentId: ctx.agentId, groupId, project, issueId: issue.iid,
|
|
issueTitle: issue.title, issueDescription: issue.description ?? "", issueUrl: issue.web_url,
|
|
role, tier: selectedTier, fromLabel: currentLabel, toLabel: targetLabel,
|
|
transitionLabel: (id, from, to) => provider.transitionLabel(id, from as StateLabel, to as StateLabel),
|
|
pluginConfig, sessionKey: ctx.sessionKey,
|
|
});
|
|
|
|
// Notify
|
|
const notifyConfig = getNotificationConfig(pluginConfig);
|
|
await notify(
|
|
{ type: "workerStart", project: project.name, groupId, issueId: issue.iid, issueTitle: issue.title, issueUrl: issue.web_url, role, tier: dr.tier, sessionAction: dr.sessionAction },
|
|
{ workspaceDir, config: notifyConfig, groupId, channel: context.channel },
|
|
);
|
|
|
|
// Tick: fill parallel slots
|
|
let tickResult: TickResult | null = null;
|
|
try {
|
|
tickResult = await projectTick({
|
|
workspaceDir, groupId, agentId: ctx.agentId, pluginConfig, sessionKey: ctx.sessionKey,
|
|
targetRole: role === "dev" ? "qa" : "dev",
|
|
});
|
|
} catch { /* non-fatal */ }
|
|
|
|
const output: Record<string, unknown> = {
|
|
success: true, project: project.name, groupId, issueId: issue.iid, issueTitle: issue.title,
|
|
role, tier: dr.tier, model: dr.model, sessionAction: dr.sessionAction,
|
|
announcement: dr.announcement, labelTransition: `${currentLabel} → ${targetLabel}`,
|
|
tierReason, tierSource,
|
|
autoDetected: { projectGroupId: !groupIdParam, role: !roleParam, issueId: issueIdParam === undefined, tier: !tierParam },
|
|
};
|
|
if (tickResult?.pickups.length) output.tickPickups = tickResult.pickups;
|
|
|
|
return jsonResult(output);
|
|
},
|
|
});
|
|
}
|