mirror of
https://github.com/alkimake/paperclip.git
synced 2026-06-16 10:50:38 +09:00
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies > - Reliable execution depends on heartbeat routing, issue lifecycle semantics, telemetry, and a fast enough local verification loop to keep regressions visible > - The remaining commits on this branch were mostly server/runtime correctness fixes plus test and documentation follow-ups in that area > - Those changes are logically separate from the UI-focused issue-detail and workspace/navigation branches even when they touch overlapping issue APIs > - This pull request groups the execution reliability, heartbeat, telemetry, and tooling changes into one standalone branch > - The benefit is a focused review of the control-plane correctness work, including the follow-up fix that restored the implicit comment-reopen helpers after branch splitting ## What Changed - Hardened issue/heartbeat execution behavior, including self-review stage skipping, deferred mention wakes during active execution, stranded execution recovery, active-run scoping, assignee resolution, and blocked-to-todo wake resumption - Reduced noisy polling/logging overhead by trimming issue run payloads, compacting persisted run logs, silencing high-volume request logs, and capping heartbeat-run queries in dashboard/inbox surfaces - Expanded telemetry and status semantics with adapter/model fields on task completion plus clearer status guidance in docs/onboarding material - Updated test infrastructure and verification defaults with faster route-test module isolation, cheaper default `pnpm test`, e2e isolation from local state, and repo verification follow-ups - Included docs/release housekeeping from the branch and added a small follow-up commit restoring the implicit comment-reopen helpers that were dropped during branch reconstruction ## Verification - `pnpm vitest run server/src/__tests__/issue-comment-reopen-routes.test.ts server/src/__tests__/issue-telemetry-routes.test.ts` - `pnpm vitest run server/src/__tests__/http-log-policy.test.ts server/src/__tests__/heartbeat-run-log.test.ts server/src/__tests__/health.test.ts` - `server/src/__tests__/activity-service.test.ts`, `server/src/__tests__/heartbeat-comment-wake-batching.test.ts`, and `server/src/__tests__/heartbeat-process-recovery.test.ts` were attempted on this host but the embedded Postgres harness reported init-script/data-dir problems and skipped or failed to start, so they are noted as environment-limited ## Risks - Medium: this branch changes core issue/heartbeat routing and reopen/wakeup behavior, so regressions would affect agent execution flow rather than isolated UI polish - Because it also updates verification infrastructure, reviewers should pay attention to whether the new tests are asserting the right failure modes and not just reshaping harness behavior ## Model Used - OpenAI Codex coding agent (GPT-5-class runtime in Codex CLI; exact deployed model ID is not exposed in this environment), reasoning enabled, tool use and local code execution enabled ## Checklist - [x] I have included a thinking path that traces from project context to this change - [x] I have specified the model used (with version and capability details) - [ ] I have run tests locally and they pass - [x] I have added or updated tests where applicable - [ ] If this change affects the UI, I have included before/after screenshots - [x] I have updated relevant documentation to reflect my changes - [x] I have considered and documented any risks above - [x] I will address all Greptile and reviewer comments before requesting merge --------- Co-authored-by: Paperclip <noreply@paperclip.ing>
546 lines
19 KiB
TypeScript
546 lines
19 KiB
TypeScript
import { randomUUID } from "node:crypto";
|
|
import type { IssueExecutionDecision, IssueExecutionPolicy, IssueExecutionStage, IssueExecutionStagePrincipal, IssueExecutionState } from "@paperclipai/shared";
|
|
import { issueExecutionPolicySchema, issueExecutionStateSchema } from "@paperclipai/shared";
|
|
import { unprocessable } from "../errors.js";
|
|
|
|
type AssigneeLike = {
|
|
assigneeAgentId?: string | null;
|
|
assigneeUserId?: string | null;
|
|
};
|
|
|
|
type IssueLike = AssigneeLike & {
|
|
status: string;
|
|
executionPolicy?: IssueExecutionPolicy | Record<string, unknown> | null;
|
|
executionState?: IssueExecutionState | Record<string, unknown> | null;
|
|
};
|
|
|
|
type ActorLike = {
|
|
agentId?: string | null;
|
|
userId?: string | null;
|
|
};
|
|
|
|
type RequestedAssigneePatch = {
|
|
assigneeAgentId?: string | null;
|
|
assigneeUserId?: string | null;
|
|
};
|
|
|
|
type TransitionInput = {
|
|
issue: IssueLike;
|
|
policy: IssueExecutionPolicy | null;
|
|
requestedStatus?: string;
|
|
requestedAssigneePatch: RequestedAssigneePatch;
|
|
actor: ActorLike;
|
|
commentBody?: string | null;
|
|
};
|
|
|
|
type TransitionResult = {
|
|
patch: Record<string, unknown>;
|
|
decision?: Pick<IssueExecutionDecision, "stageId" | "stageType" | "outcome" | "body">;
|
|
workflowControlledAssignment?: boolean;
|
|
};
|
|
|
|
const COMPLETED_STATUS: IssueExecutionState["status"] = "completed";
|
|
const PENDING_STATUS: IssueExecutionState["status"] = "pending";
|
|
const CHANGES_REQUESTED_STATUS: IssueExecutionState["status"] = "changes_requested";
|
|
|
|
export function normalizeIssueExecutionPolicy(input: unknown): IssueExecutionPolicy | null {
|
|
if (input == null) return null;
|
|
const parsed = issueExecutionPolicySchema.safeParse(input);
|
|
if (!parsed.success) {
|
|
throw unprocessable("Invalid execution policy", parsed.error.flatten());
|
|
}
|
|
|
|
const stages = parsed.data.stages
|
|
.map((stage) => {
|
|
const participants: IssueExecutionStage["participants"] = stage.participants
|
|
.map((participant) => ({
|
|
id: participant.id ?? randomUUID(),
|
|
type: participant.type,
|
|
agentId: participant.type === "agent" ? participant.agentId ?? null : null,
|
|
userId: participant.type === "user" ? participant.userId ?? null : null,
|
|
}))
|
|
.filter((participant) => (participant.type === "agent" ? Boolean(participant.agentId) : Boolean(participant.userId)));
|
|
|
|
const dedupedParticipants: IssueExecutionStage["participants"] = [];
|
|
const seen = new Set<string>();
|
|
for (const participant of participants) {
|
|
const key = participant.type === "agent" ? `agent:${participant.agentId}` : `user:${participant.userId}`;
|
|
if (seen.has(key)) continue;
|
|
seen.add(key);
|
|
dedupedParticipants.push(participant);
|
|
}
|
|
|
|
if (dedupedParticipants.length === 0) return null;
|
|
return {
|
|
id: stage.id ?? randomUUID(),
|
|
type: stage.type,
|
|
approvalsNeeded: 1 as const,
|
|
participants: dedupedParticipants,
|
|
};
|
|
})
|
|
.filter((stage): stage is NonNullable<typeof stage> => stage !== null);
|
|
|
|
if (stages.length === 0) return null;
|
|
|
|
return {
|
|
mode: parsed.data.mode ?? "normal",
|
|
commentRequired: true,
|
|
stages,
|
|
};
|
|
}
|
|
|
|
export function parseIssueExecutionState(input: unknown): IssueExecutionState | null {
|
|
if (input == null) return null;
|
|
const parsed = issueExecutionStateSchema.safeParse(input);
|
|
if (!parsed.success) return null;
|
|
return parsed.data;
|
|
}
|
|
|
|
export function assigneePrincipal(input: AssigneeLike): IssueExecutionStagePrincipal | null {
|
|
if (input.assigneeAgentId) {
|
|
return { type: "agent", agentId: input.assigneeAgentId, userId: null };
|
|
}
|
|
if (input.assigneeUserId) {
|
|
return { type: "user", userId: input.assigneeUserId, agentId: null };
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function actorPrincipal(actor: ActorLike): IssueExecutionStagePrincipal | null {
|
|
if (actor.agentId) return { type: "agent", agentId: actor.agentId, userId: null };
|
|
if (actor.userId) return { type: "user", userId: actor.userId, agentId: null };
|
|
return null;
|
|
}
|
|
|
|
function principalsEqual(a: IssueExecutionStagePrincipal | null, b: IssueExecutionStagePrincipal | null): boolean {
|
|
if (!a || !b) return false;
|
|
if (a.type !== b.type) return false;
|
|
return a.type === "agent" ? a.agentId === b.agentId : a.userId === b.userId;
|
|
}
|
|
|
|
function findStageById(policy: IssueExecutionPolicy, stageId: string | null | undefined) {
|
|
if (!stageId) return null;
|
|
return policy.stages.find((stage) => stage.id === stageId) ?? null;
|
|
}
|
|
|
|
function nextPendingStage(policy: IssueExecutionPolicy, state: IssueExecutionState | null) {
|
|
const completed = new Set(state?.completedStageIds ?? []);
|
|
return policy.stages.find((stage) => !completed.has(stage.id)) ?? null;
|
|
}
|
|
|
|
function selectStageParticipant(
|
|
stage: IssueExecutionStage,
|
|
opts?: {
|
|
preferred?: IssueExecutionStagePrincipal | null;
|
|
exclude?: IssueExecutionStagePrincipal | null;
|
|
},
|
|
): IssueExecutionStagePrincipal | null {
|
|
const participants = stage.participants.filter((participant) => !principalsEqual(participant, opts?.exclude ?? null));
|
|
if (participants.length === 0) return null;
|
|
if (opts?.preferred) {
|
|
const preferred = participants.find((participant) => principalsEqual(participant, opts.preferred ?? null));
|
|
if (preferred) return preferred;
|
|
}
|
|
const first = participants[0];
|
|
return first ? { type: first.type, agentId: first.agentId ?? null, userId: first.userId ?? null } : null;
|
|
}
|
|
|
|
function stageHasParticipant(stage: IssueExecutionStage, participant: IssueExecutionStagePrincipal | null): boolean {
|
|
if (!participant) return false;
|
|
return stage.participants.some((candidate) => principalsEqual(candidate, participant));
|
|
}
|
|
|
|
function patchForPrincipal(principal: IssueExecutionStagePrincipal | null) {
|
|
if (!principal) {
|
|
return { assigneeAgentId: null, assigneeUserId: null };
|
|
}
|
|
return principal.type === "agent"
|
|
? { assigneeAgentId: principal.agentId ?? null, assigneeUserId: null }
|
|
: { assigneeAgentId: null, assigneeUserId: principal.userId ?? null };
|
|
}
|
|
|
|
function buildCompletedState(previous: IssueExecutionState | null, currentStage: IssueExecutionStage): IssueExecutionState {
|
|
const completedStageIds = Array.from(new Set([...(previous?.completedStageIds ?? []), currentStage.id]));
|
|
return {
|
|
status: COMPLETED_STATUS,
|
|
currentStageId: null,
|
|
currentStageIndex: null,
|
|
currentStageType: null,
|
|
currentParticipant: null,
|
|
returnAssignee: previous?.returnAssignee ?? null,
|
|
completedStageIds,
|
|
lastDecisionId: previous?.lastDecisionId ?? null,
|
|
lastDecisionOutcome: "approved",
|
|
};
|
|
}
|
|
|
|
function buildStateWithCompletedStages(input: {
|
|
previous: IssueExecutionState | null;
|
|
completedStageIds: string[];
|
|
returnAssignee: IssueExecutionStagePrincipal | null;
|
|
}): IssueExecutionState {
|
|
return {
|
|
status: input.previous?.status ?? PENDING_STATUS,
|
|
currentStageId: input.previous?.currentStageId ?? null,
|
|
currentStageIndex: input.previous?.currentStageIndex ?? null,
|
|
currentStageType: input.previous?.currentStageType ?? null,
|
|
currentParticipant: input.previous?.currentParticipant ?? null,
|
|
returnAssignee: input.previous?.returnAssignee ?? input.returnAssignee,
|
|
completedStageIds: input.completedStageIds,
|
|
lastDecisionId: input.previous?.lastDecisionId ?? null,
|
|
lastDecisionOutcome: input.previous?.lastDecisionOutcome ?? null,
|
|
};
|
|
}
|
|
|
|
function buildSkippedStageCompletedState(input: {
|
|
previous: IssueExecutionState | null;
|
|
completedStageIds: string[];
|
|
returnAssignee: IssueExecutionStagePrincipal | null;
|
|
}): IssueExecutionState {
|
|
return {
|
|
status: COMPLETED_STATUS,
|
|
currentStageId: null,
|
|
currentStageIndex: null,
|
|
currentStageType: null,
|
|
currentParticipant: null,
|
|
returnAssignee: input.previous?.returnAssignee ?? input.returnAssignee,
|
|
completedStageIds: input.completedStageIds,
|
|
lastDecisionId: input.previous?.lastDecisionId ?? null,
|
|
lastDecisionOutcome: input.previous?.lastDecisionOutcome ?? null,
|
|
};
|
|
}
|
|
|
|
function buildPendingState(input: {
|
|
previous: IssueExecutionState | null;
|
|
stage: IssueExecutionStage;
|
|
stageIndex: number;
|
|
participant: IssueExecutionStagePrincipal;
|
|
returnAssignee: IssueExecutionStagePrincipal | null;
|
|
}): IssueExecutionState {
|
|
return {
|
|
status: PENDING_STATUS,
|
|
currentStageId: input.stage.id,
|
|
currentStageIndex: input.stageIndex,
|
|
currentStageType: input.stage.type,
|
|
currentParticipant: input.participant,
|
|
returnAssignee: input.returnAssignee,
|
|
completedStageIds: input.previous?.completedStageIds ?? [],
|
|
lastDecisionId: input.previous?.lastDecisionId ?? null,
|
|
lastDecisionOutcome: input.previous?.lastDecisionOutcome ?? null,
|
|
};
|
|
}
|
|
|
|
function buildChangesRequestedState(previous: IssueExecutionState, currentStage: IssueExecutionStage): IssueExecutionState {
|
|
return {
|
|
...previous,
|
|
status: CHANGES_REQUESTED_STATUS,
|
|
currentStageId: currentStage.id,
|
|
currentStageType: currentStage.type,
|
|
lastDecisionOutcome: "changes_requested",
|
|
};
|
|
}
|
|
|
|
function buildPendingStagePatch(input: {
|
|
patch: Record<string, unknown>;
|
|
previous: IssueExecutionState | null;
|
|
policy: IssueExecutionPolicy;
|
|
stage: IssueExecutionStage;
|
|
participant: IssueExecutionStagePrincipal;
|
|
returnAssignee: IssueExecutionStagePrincipal | null;
|
|
}) {
|
|
input.patch.status = "in_review";
|
|
Object.assign(input.patch, patchForPrincipal(input.participant));
|
|
input.patch.executionState = buildPendingState({
|
|
previous: input.previous,
|
|
stage: input.stage,
|
|
stageIndex: input.policy.stages.findIndex((candidate) => candidate.id === input.stage.id),
|
|
participant: input.participant,
|
|
returnAssignee: input.returnAssignee,
|
|
});
|
|
}
|
|
|
|
function clearExecutionStatePatch(input: {
|
|
patch: Record<string, unknown>;
|
|
issueStatus: string;
|
|
requestedStatus?: string;
|
|
returnAssignee: IssueExecutionStagePrincipal | null;
|
|
}) {
|
|
input.patch.executionState = null;
|
|
if (input.requestedStatus === undefined && input.issueStatus === "in_review" && input.returnAssignee) {
|
|
input.patch.status = "in_progress";
|
|
Object.assign(input.patch, patchForPrincipal(input.returnAssignee));
|
|
}
|
|
}
|
|
|
|
function canAutoSkipPendingStage(input: {
|
|
stage: IssueExecutionStage;
|
|
returnAssignee: IssueExecutionStagePrincipal | null;
|
|
requestedStatus?: string;
|
|
}) {
|
|
if (input.requestedStatus !== "done" || input.stage.type !== "review" || !input.returnAssignee) {
|
|
return false;
|
|
}
|
|
return input.stage.participants.length > 0 &&
|
|
input.stage.participants.every((participant) => principalsEqual(participant, input.returnAssignee));
|
|
}
|
|
|
|
export function applyIssueExecutionPolicyTransition(input: TransitionInput): TransitionResult {
|
|
const patch: Record<string, unknown> = {};
|
|
const existingState = parseIssueExecutionState(input.issue.executionState);
|
|
const currentAssignee = assigneePrincipal(input.issue);
|
|
const actor = actorPrincipal(input.actor);
|
|
const requestedAssigneePatchProvided =
|
|
input.requestedAssigneePatch.assigneeAgentId !== undefined || input.requestedAssigneePatch.assigneeUserId !== undefined;
|
|
const explicitAssignee = assigneePrincipal(input.requestedAssigneePatch);
|
|
const currentStage = input.policy ? findStageById(input.policy, existingState?.currentStageId) : null;
|
|
const requestedStatus = input.requestedStatus;
|
|
const activeStage = currentStage && existingState?.status === PENDING_STATUS ? currentStage : null;
|
|
|
|
if (!input.policy) {
|
|
if (existingState) {
|
|
patch.executionState = null;
|
|
if (input.issue.status === "in_review" && existingState.returnAssignee) {
|
|
patch.status = "in_progress";
|
|
Object.assign(patch, patchForPrincipal(existingState.returnAssignee));
|
|
}
|
|
}
|
|
return { patch };
|
|
}
|
|
|
|
if (
|
|
(input.issue.status === "done" || input.issue.status === "cancelled") &&
|
|
requestedStatus &&
|
|
requestedStatus !== "done" &&
|
|
requestedStatus !== "cancelled"
|
|
) {
|
|
patch.executionState = null;
|
|
return { patch };
|
|
}
|
|
|
|
if (existingState?.currentStageId && !currentStage) {
|
|
clearExecutionStatePatch({
|
|
patch,
|
|
issueStatus: input.issue.status,
|
|
requestedStatus,
|
|
returnAssignee: existingState.returnAssignee,
|
|
});
|
|
return { patch };
|
|
}
|
|
|
|
if (activeStage) {
|
|
const currentParticipant =
|
|
existingState?.currentParticipant ??
|
|
selectStageParticipant(activeStage, {
|
|
exclude: existingState?.returnAssignee ?? null,
|
|
});
|
|
if (!currentParticipant) {
|
|
throw unprocessable(`No eligible ${activeStage.type} participant is configured for this issue`);
|
|
}
|
|
|
|
if (!stageHasParticipant(activeStage, currentParticipant)) {
|
|
const participant = selectStageParticipant(activeStage, {
|
|
preferred: explicitAssignee ?? existingState?.currentParticipant ?? null,
|
|
exclude: existingState?.returnAssignee ?? null,
|
|
});
|
|
if (!participant) {
|
|
clearExecutionStatePatch({
|
|
patch,
|
|
issueStatus: input.issue.status,
|
|
requestedStatus,
|
|
returnAssignee: existingState?.returnAssignee ?? null,
|
|
});
|
|
return { patch };
|
|
}
|
|
|
|
buildPendingStagePatch({
|
|
patch,
|
|
previous: existingState,
|
|
policy: input.policy,
|
|
stage: activeStage,
|
|
participant,
|
|
returnAssignee: existingState?.returnAssignee ?? currentAssignee ?? actor,
|
|
});
|
|
return {
|
|
patch,
|
|
workflowControlledAssignment: true,
|
|
};
|
|
}
|
|
|
|
if (principalsEqual(currentParticipant, actor)) {
|
|
if (requestedStatus === "done") {
|
|
if (!input.commentBody?.trim()) {
|
|
throw unprocessable("Approving a review or approval stage requires a comment");
|
|
}
|
|
const approvedState = buildCompletedState(existingState, activeStage);
|
|
const nextStage = nextPendingStage(
|
|
input.policy,
|
|
{ ...approvedState, completedStageIds: approvedState.completedStageIds },
|
|
);
|
|
|
|
if (!nextStage) {
|
|
patch.executionState = approvedState;
|
|
return {
|
|
patch,
|
|
decision: {
|
|
stageId: activeStage.id,
|
|
stageType: activeStage.type,
|
|
outcome: "approved",
|
|
body: input.commentBody.trim(),
|
|
},
|
|
};
|
|
}
|
|
|
|
const participant = selectStageParticipant(nextStage, {
|
|
preferred: explicitAssignee,
|
|
exclude: existingState?.returnAssignee ?? null,
|
|
});
|
|
if (!participant) {
|
|
throw unprocessable(`No eligible ${nextStage.type} participant is configured for this issue`);
|
|
}
|
|
|
|
buildPendingStagePatch({
|
|
patch,
|
|
previous: approvedState,
|
|
policy: input.policy,
|
|
stage: nextStage,
|
|
participant,
|
|
returnAssignee: existingState?.returnAssignee ?? currentAssignee ?? actor,
|
|
});
|
|
return {
|
|
patch,
|
|
decision: {
|
|
stageId: activeStage.id,
|
|
stageType: activeStage.type,
|
|
outcome: "approved",
|
|
body: input.commentBody.trim(),
|
|
},
|
|
workflowControlledAssignment: true,
|
|
};
|
|
}
|
|
|
|
if (requestedStatus && requestedStatus !== "in_review") {
|
|
if (!input.commentBody?.trim()) {
|
|
throw unprocessable("Requesting changes requires a comment");
|
|
}
|
|
if (!existingState?.returnAssignee) {
|
|
throw unprocessable("This execution stage has no return assignee");
|
|
}
|
|
patch.status = "in_progress";
|
|
Object.assign(patch, patchForPrincipal(existingState.returnAssignee));
|
|
patch.executionState = buildChangesRequestedState(existingState, activeStage);
|
|
return {
|
|
patch,
|
|
decision: {
|
|
stageId: activeStage.id,
|
|
stageType: activeStage.type,
|
|
outcome: "changes_requested",
|
|
body: input.commentBody.trim(),
|
|
},
|
|
workflowControlledAssignment: true,
|
|
};
|
|
}
|
|
}
|
|
|
|
const attemptedStageAdvance =
|
|
(requestedStatus !== undefined && requestedStatus !== "in_review") ||
|
|
(requestedAssigneePatchProvided && !principalsEqual(explicitAssignee, currentParticipant));
|
|
const stageStateDrifted =
|
|
input.issue.status !== "in_review" ||
|
|
!principalsEqual(currentAssignee, currentParticipant) ||
|
|
!principalsEqual(existingState?.currentParticipant ?? null, currentParticipant);
|
|
|
|
if (attemptedStageAdvance && !stageStateDrifted) {
|
|
throw unprocessable("Only the active reviewer or approver can advance the current execution stage");
|
|
}
|
|
|
|
if (stageStateDrifted) {
|
|
buildPendingStagePatch({
|
|
patch,
|
|
previous: existingState,
|
|
policy: input.policy,
|
|
stage: activeStage,
|
|
participant: currentParticipant,
|
|
returnAssignee: existingState?.returnAssignee ?? currentAssignee ?? actor,
|
|
});
|
|
return {
|
|
patch,
|
|
workflowControlledAssignment: true,
|
|
};
|
|
}
|
|
|
|
return { patch };
|
|
}
|
|
|
|
const shouldStartWorkflow =
|
|
requestedStatus === "done" ||
|
|
requestedStatus === "in_review";
|
|
|
|
if (!shouldStartWorkflow) {
|
|
return { patch };
|
|
}
|
|
|
|
let pendingStage =
|
|
existingState?.status === CHANGES_REQUESTED_STATUS && currentStage
|
|
? currentStage
|
|
: nextPendingStage(input.policy, existingState);
|
|
if (!pendingStage) return { patch };
|
|
|
|
const returnAssignee = existingState?.returnAssignee ?? currentAssignee;
|
|
const skippedStageIds = [...(existingState?.completedStageIds ?? [])];
|
|
let participant = selectStageParticipant(pendingStage, {
|
|
preferred:
|
|
existingState?.status === CHANGES_REQUESTED_STATUS
|
|
? explicitAssignee ?? existingState.currentParticipant ?? null
|
|
: explicitAssignee,
|
|
exclude: returnAssignee,
|
|
});
|
|
while (!participant && canAutoSkipPendingStage({ stage: pendingStage, returnAssignee, requestedStatus })) {
|
|
skippedStageIds.push(pendingStage.id);
|
|
pendingStage = nextPendingStage(
|
|
input.policy,
|
|
buildStateWithCompletedStages({
|
|
previous: existingState,
|
|
completedStageIds: skippedStageIds,
|
|
returnAssignee,
|
|
}),
|
|
);
|
|
if (!pendingStage) {
|
|
patch.executionState = buildSkippedStageCompletedState({
|
|
previous: existingState,
|
|
completedStageIds: skippedStageIds,
|
|
returnAssignee,
|
|
});
|
|
return { patch };
|
|
}
|
|
participant = selectStageParticipant(pendingStage, {
|
|
preferred:
|
|
existingState?.status === CHANGES_REQUESTED_STATUS
|
|
? explicitAssignee ?? existingState.currentParticipant ?? null
|
|
: explicitAssignee,
|
|
exclude: returnAssignee,
|
|
});
|
|
}
|
|
if (!participant) {
|
|
throw unprocessable(`No eligible ${pendingStage.type} participant is configured for this issue`);
|
|
}
|
|
|
|
buildPendingStagePatch({
|
|
patch,
|
|
previous:
|
|
skippedStageIds.length === (existingState?.completedStageIds ?? []).length
|
|
? existingState
|
|
: buildStateWithCompletedStages({
|
|
previous: existingState,
|
|
completedStageIds: skippedStageIds,
|
|
returnAssignee,
|
|
}),
|
|
policy: input.policy,
|
|
stage: pendingStage,
|
|
participant,
|
|
returnAssignee,
|
|
});
|
|
return {
|
|
patch,
|
|
workflowControlledAssignment: true,
|
|
};
|
|
}
|