0808b388ee
## Thinking Path > - Paperclip is a control plane for autonomous AI companies, where work must end with a clear disposition rather than ambiguous agent liveness. > - Recovery currently detects stalled or missing-next-step issues, but source issue recovery can become split across child recovery issues, blockers, and comments. > - That makes it harder for operators and agents to see who owns recovery and what exact action is needed on the original issue. > - Source-scoped recovery actions give the original issue a first-class active recovery state with owner, evidence, wake policy, and resolution outcome. > - This pull request adds the recovery-action data model, backend reconciliation and resolution APIs, and board UI indicators/actions. > - The benefit is clearer stalled-work recovery without losing source issue context or relying on comments as the liveness path. ## What Changed - Added the `issue_recovery_actions` schema, shared types/constants/validators, and an idempotent `0084_issue_recovery_actions` migration ordered after current `master` migrations. - Updated stranded/missing-disposition recovery to create source-scoped recovery actions, wake the recovery owner on the source issue, and avoid locking the source issue for recovery-action wakes. - Added API support for reading active recovery actions on issue detail/list surfaces and resolving them with restored, blocked, cancelled, or false-positive outcomes. - Require blocked recovery resolutions to have an unresolved first-class blocker, and removed the UI shortcut that could mark recovery blocked without a blocker selection path. - Surfaced recovery indicators/actions in the issue UI, blocker notices, active run panels, issue rows, and Storybook coverage. - Updated docs and focused tests for recovery semantics, ownership, races, stale comments, and UI behavior. ## Verification - `pnpm exec vitest run server/src/__tests__/issue-recovery-actions.test.ts server/src/__tests__/heartbeat-process-recovery.test.ts ui/src/components/IssueRecoveryActionCard.test.tsx ui/src/components/IssueBlockedNotice.test.tsx ui/src/api/issues.test.ts` — 5 files, 72 tests passed. - `pnpm --filter @paperclipai/shared typecheck` — passed. - `pnpm --filter @paperclipai/db typecheck` — passed, including migration numbering check. - `pnpm --filter @paperclipai/server typecheck` — passed. - `pnpm --filter @paperclipai/ui typecheck` — passed. - Follow-up verification after blocker-resolution guard: `pnpm exec vitest run server/src/__tests__/issue-recovery-actions.test.ts ui/src/components/IssueRecoveryActionCard.test.tsx ui/src/api/issues.test.ts` — 3 files, 27 tests passed. - Follow-up `pnpm --filter @paperclipai/server typecheck` — passed. - Follow-up `pnpm --filter @paperclipai/ui typecheck` — passed. - UI states are available in `ui/storybook/stories/source-issue-recovery.stories.tsx`; screenshot capture helper is `scripts/screenshot-recovery-card.cjs`. ## Risks - Medium: recovery behavior changes from child recovery issue ownership toward source-scoped actions, so operators may see stalled-work state in new places. - Migration risk is mitigated by using the next migration slot after `master` and making the table/constraints/index creation idempotent for anyone who previously applied the old branch-local `0082_dizzy_master_mold` migration. - Existing child recovery issue paths are still guarded for already-created recovery issues, but new source-scoped flows should be watched in CI and Greptile review. > For core feature work, check [`ROADMAP.md`](ROADMAP.md) first and discuss it in `#dev` before opening the PR. Feature PRs that overlap with planned core work may need to be redirected — check the roadmap first. See `CONTRIBUTING.md`. ## Model Used - OpenAI Codex, GPT-5 coding agent, tool use enabled for shell, Git, GitHub, and local test execution. Context window not exposed by the runtime. ## 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) - [x] I have checked ROADMAP.md and confirmed this PR does not duplicate planned core work - [x] I have run tests locally and they pass - [x] I have added or updated tests where applicable - [x] 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>
382 lines
12 KiB
TypeScript
382 lines
12 KiB
TypeScript
import type { Meta, StoryObj } from "@storybook/react-vite";
|
|
import type { ReactNode } from "react";
|
|
import type { IssueRecoveryAction, IssueRelationIssueSummary } from "@paperclipai/shared";
|
|
import { Eye, ExternalLink, OctagonAlert, RefreshCw, TriangleAlert } from "lucide-react";
|
|
import { IssueRecoveryActionCard } from "@/components/IssueRecoveryActionCard";
|
|
import { IssueRow } from "@/components/IssueRow";
|
|
import { IssueBlockedNotice } from "@/components/IssueBlockedNotice";
|
|
import { storybookAgentMap, storybookAgents, createIssue } from "../fixtures/paperclipData";
|
|
|
|
const claudeAgent = storybookAgents.find((agent) => agent.name.toLowerCase().startsWith("claude")) ?? storybookAgents[0]!;
|
|
const codexAgent = storybookAgents.find((agent) => agent.name.toLowerCase().startsWith("codex")) ?? storybookAgents[0]!;
|
|
|
|
function StoryFrame({ title, description, children }: { title: string; description?: string; children: ReactNode }) {
|
|
return (
|
|
<main className="min-h-screen bg-background p-4 text-foreground sm:p-8">
|
|
<div className="mx-auto max-w-5xl space-y-5">
|
|
<header>
|
|
<div className="text-xs font-medium uppercase tracking-[0.2em] text-muted-foreground">
|
|
Source-issue recovery
|
|
</div>
|
|
<h1 className="mt-1 text-2xl font-semibold">{title}</h1>
|
|
{description ? (
|
|
<p className="mt-2 max-w-3xl text-sm text-muted-foreground">{description}</p>
|
|
) : null}
|
|
</header>
|
|
{children}
|
|
</div>
|
|
</main>
|
|
);
|
|
}
|
|
|
|
function buildAction(overrides: Partial<IssueRecoveryAction> = {}): IssueRecoveryAction {
|
|
return {
|
|
id: "00000000-0000-0000-0000-0000000000aa",
|
|
companyId: "company-storybook",
|
|
sourceIssueId: "00000000-0000-0000-0000-0000000000ff",
|
|
recoveryIssueId: null,
|
|
kind: "missing_disposition",
|
|
status: "active",
|
|
ownerType: "agent",
|
|
ownerAgentId: claudeAgent.id,
|
|
ownerUserId: null,
|
|
previousOwnerAgentId: codexAgent.id,
|
|
returnOwnerAgentId: codexAgent.id,
|
|
cause: "missing_disposition",
|
|
fingerprint: "fp",
|
|
evidence: {
|
|
summary: "Run finished without picking a disposition. The PR has tests passing on CI.",
|
|
sourceRunId: "7accd7a4-c9ca-4db2-9233-3228a037cc09",
|
|
correctiveRunId: "2606404d-3859-4142-ba37-3228a037cc09",
|
|
},
|
|
nextAction: "Choose and record a valid issue disposition without copying transcript content.",
|
|
wakePolicy: { type: "wake_owner" },
|
|
monitorPolicy: null,
|
|
attemptCount: 1,
|
|
maxAttempts: 3,
|
|
timeoutAt: null,
|
|
lastAttemptAt: "2026-04-20T11:55:00.000Z",
|
|
outcome: null,
|
|
resolutionNote: null,
|
|
resolvedAt: null,
|
|
createdAt: "2026-04-20T11:55:00.000Z",
|
|
updatedAt: "2026-04-20T11:55:00.000Z",
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function CardPanel({ caption, action, forcedState, canFalsePositive }: {
|
|
caption: string;
|
|
action: IssueRecoveryAction;
|
|
forcedState?: React.ComponentProps<typeof IssueRecoveryActionCard>["forcedState"];
|
|
canFalsePositive?: boolean;
|
|
}) {
|
|
return (
|
|
<section className="space-y-2">
|
|
<div className="text-[11px] font-semibold uppercase tracking-[0.16em] text-muted-foreground">
|
|
{caption}
|
|
</div>
|
|
<IssueRecoveryActionCard
|
|
action={action}
|
|
agentMap={storybookAgentMap}
|
|
forcedState={forcedState}
|
|
onResolve={() => {}}
|
|
canFalsePositive={canFalsePositive}
|
|
/>
|
|
</section>
|
|
);
|
|
}
|
|
|
|
function AllStatesPanel() {
|
|
return (
|
|
<div className="grid gap-5 lg:grid-cols-1">
|
|
<CardPanel caption="State 1 · Recovery needed (default)" action={buildAction()} canFalsePositive />
|
|
<CardPanel
|
|
caption="State 2 · Recovery in progress"
|
|
action={buildAction({ outcome: "delegated", attemptCount: 2 })}
|
|
forcedState="in_progress"
|
|
canFalsePositive
|
|
/>
|
|
<CardPanel
|
|
caption="State 3 · Observing active run (watchdog)"
|
|
action={buildAction({
|
|
kind: "active_run_watchdog",
|
|
wakePolicy: { type: "monitor", intervalLabel: "in 4m" },
|
|
evidence: { summary: "The active run has been silent for 7 minutes. Last log: 'continuing checks…'" },
|
|
nextAction: "Observe the active run; intervene only if the silence persists past timeout.",
|
|
})}
|
|
/>
|
|
<CardPanel
|
|
caption="State 4 · Recovery escalated"
|
|
action={buildAction({
|
|
status: "escalated",
|
|
attemptCount: 3,
|
|
wakePolicy: { type: "board_escalation" },
|
|
evidence: {
|
|
summary: "Three corrective wakes failed. The recovery owner has not produced a disposition.",
|
|
sourceRunId: "7accd7a4-c9ca-4db2-9233-3228a037cc09",
|
|
},
|
|
nextAction: "Board operator: assign an invokable owner or record a manual resolution.",
|
|
})}
|
|
canFalsePositive
|
|
/>
|
|
<CardPanel
|
|
caption="State 5 · Recovery resolved"
|
|
action={buildAction({
|
|
status: "resolved",
|
|
outcome: "restored",
|
|
resolvedAt: "2026-04-20T12:01:00.000Z",
|
|
nextAction: "Issue restored to a valid disposition.",
|
|
})}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function buildBlocker(
|
|
overrides: Partial<IssueRelationIssueSummary> = {},
|
|
): IssueRelationIssueSummary {
|
|
return {
|
|
id: "blocker-1",
|
|
identifier: "PAP-9065",
|
|
title: "Add full company search page",
|
|
status: "in_progress",
|
|
priority: "medium",
|
|
assigneeAgentId: claudeAgent.id,
|
|
assigneeUserId: null,
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function BlockerNoticePanel() {
|
|
return (
|
|
<div className="space-y-4">
|
|
<IssueBlockedNotice
|
|
issueStatus="blocked"
|
|
blockers={[
|
|
buildBlocker({ activeRecoveryAction: buildAction() }),
|
|
buildBlocker({
|
|
id: "blocker-2",
|
|
identifier: "PAP-9099",
|
|
title: "Watchdog: PR review pipeline silent",
|
|
activeRecoveryAction: buildAction({ kind: "active_run_watchdog" }),
|
|
}),
|
|
buildBlocker({
|
|
id: "blocker-3",
|
|
identifier: "PAP-9073",
|
|
title: "Recovery escalated for stranded run",
|
|
status: "blocked",
|
|
activeRecoveryAction: buildAction({ status: "escalated" }),
|
|
}),
|
|
buildBlocker({
|
|
id: "blocker-4",
|
|
identifier: "PAP-9051",
|
|
title: "Bare blocker without recovery state",
|
|
}),
|
|
]}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
type RunCardRecoveryState = "needed" | "in_progress" | "observe_only" | "escalated";
|
|
|
|
const RUN_CARD_RECOVERY_TONE: Record<RunCardRecoveryState, { icon: typeof TriangleAlert; label: string; className: string }> = {
|
|
needed: {
|
|
icon: TriangleAlert,
|
|
label: "Recovery needed",
|
|
className: "border-amber-500/60 bg-amber-500/15 text-amber-700 dark:text-amber-300",
|
|
},
|
|
in_progress: {
|
|
icon: RefreshCw,
|
|
label: "Recovery in progress",
|
|
className: "border-sky-500/60 bg-sky-500/15 text-sky-700 dark:text-sky-300",
|
|
},
|
|
observe_only: {
|
|
icon: Eye,
|
|
label: "Observing active run",
|
|
className: "border-border bg-muted text-muted-foreground",
|
|
},
|
|
escalated: {
|
|
icon: OctagonAlert,
|
|
label: "Recovery escalated",
|
|
className: "border-red-500/60 bg-red-500/15 text-red-700 dark:text-red-300",
|
|
},
|
|
};
|
|
|
|
function ActiveRunRecoveryChip({ state }: { state: RunCardRecoveryState }) {
|
|
const tone = RUN_CARD_RECOVERY_TONE[state];
|
|
const Icon = tone.icon;
|
|
return (
|
|
<span
|
|
className={`inline-flex shrink-0 items-center gap-0.5 rounded-full border px-1.5 py-0.5 text-[10px] font-medium ${tone.className}`}
|
|
role="status"
|
|
aria-label={tone.label}
|
|
>
|
|
<Icon className="h-2.5 w-2.5" aria-hidden />
|
|
{tone.label}
|
|
</span>
|
|
);
|
|
}
|
|
|
|
function ActiveRunCardMock({
|
|
identifier,
|
|
title,
|
|
recoveryState,
|
|
}: {
|
|
identifier: string;
|
|
title: string;
|
|
recoveryState: RunCardRecoveryState;
|
|
}) {
|
|
return (
|
|
<div className="flex h-[260px] w-full max-w-[320px] flex-col overflow-hidden rounded-xl border border-cyan-500/25 bg-cyan-500/[0.04] shadow-[0_16px_40px_rgba(6,182,212,0.08)]">
|
|
<div className="border-b border-border/60 px-3 py-3">
|
|
<div className="flex items-start justify-between gap-2">
|
|
<div className="min-w-0">
|
|
<div className="flex items-center gap-2">
|
|
<span className="relative flex h-2.5 w-2.5 shrink-0">
|
|
<span className="absolute inline-flex h-full w-full animate-ping rounded-full bg-cyan-400 opacity-70" />
|
|
<span className="relative inline-flex h-2.5 w-2.5 rounded-full bg-cyan-500" />
|
|
</span>
|
|
<span className="text-sm font-medium">CodexCoder</span>
|
|
</div>
|
|
<div className="mt-2 flex items-center gap-2 text-[11px] text-muted-foreground">
|
|
<span>Live now</span>
|
|
</div>
|
|
</div>
|
|
<span className="inline-flex items-center gap-1 rounded-full border border-border/70 bg-background/70 px-2 py-1 text-[10px] text-muted-foreground">
|
|
<ExternalLink className="h-2.5 w-2.5" />
|
|
</span>
|
|
</div>
|
|
<div className="mt-3 rounded-lg border border-border/60 bg-background/60 px-2.5 py-2 text-xs">
|
|
<span className="line-clamp-2 text-cyan-700 dark:text-cyan-300">
|
|
{identifier} - {title}
|
|
</span>
|
|
<div className="mt-1.5">
|
|
<ActiveRunRecoveryChip state={recoveryState} />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div className="flex-1 px-3 py-2 text-[11px] text-muted-foreground">Live transcript…</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function ActiveRunPanel() {
|
|
return (
|
|
<div className="grid gap-4 sm:grid-cols-2">
|
|
<ActiveRunCardMock
|
|
identifier="PAP-9065"
|
|
title="Add full company search page"
|
|
recoveryState="needed"
|
|
/>
|
|
<ActiveRunCardMock
|
|
identifier="PAP-9099"
|
|
title="Watchdog: PR review pipeline silent"
|
|
recoveryState="observe_only"
|
|
/>
|
|
<ActiveRunCardMock
|
|
identifier="PAP-9073"
|
|
title="Recovery escalated for stranded run"
|
|
recoveryState="escalated"
|
|
/>
|
|
<ActiveRunCardMock
|
|
identifier="PAP-9101"
|
|
title="Recovery in progress: delegated"
|
|
recoveryState="in_progress"
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function InboxRowPanel() {
|
|
const baseIssue = createIssue();
|
|
return (
|
|
<div className="rounded-lg border border-border/70 bg-background/80">
|
|
<IssueRow
|
|
issue={{
|
|
...baseIssue,
|
|
identifier: "PAP-9065",
|
|
title: "Add full company search page",
|
|
status: "in_progress",
|
|
activeRecoveryAction: buildAction(),
|
|
}}
|
|
/>
|
|
<IssueRow
|
|
issue={{
|
|
...baseIssue,
|
|
id: "issue-recovery-watch",
|
|
identifier: "PAP-9099",
|
|
title: "Watchdog: PR review pipeline silent",
|
|
status: "in_progress",
|
|
activeRecoveryAction: buildAction({ kind: "active_run_watchdog" }),
|
|
}}
|
|
/>
|
|
<IssueRow
|
|
issue={{
|
|
...baseIssue,
|
|
id: "issue-recovery-escalated",
|
|
identifier: "PAP-9073",
|
|
title: "Recovery escalated for stranded run",
|
|
status: "blocked",
|
|
activeRecoveryAction: buildAction({ status: "escalated" }),
|
|
}}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const meta = {
|
|
title: "Paperclip/Source Issue Recovery",
|
|
component: AllStatesPanel,
|
|
parameters: { layout: "fullscreen" },
|
|
} satisfies Meta<typeof AllStatesPanel>;
|
|
|
|
export default meta;
|
|
|
|
type Story = StoryObj<typeof meta>;
|
|
|
|
export const RecoveryActionCardStates: Story = {
|
|
render: () => (
|
|
<StoryFrame
|
|
title="Recovery action card states"
|
|
description="Five states required by the source-issue recovery contract: needed, in progress, observe-only watchdog, escalated, resolved."
|
|
>
|
|
<AllStatesPanel />
|
|
</StoryFrame>
|
|
),
|
|
};
|
|
|
|
export const InboxRowChips: Story = {
|
|
render: () => (
|
|
<StoryFrame
|
|
title="Inbox row recovery chips"
|
|
description="Source rows expose recovery state inline; no synthetic sibling row appears for source-scoped recovery."
|
|
>
|
|
<InboxRowPanel />
|
|
</StoryFrame>
|
|
),
|
|
};
|
|
|
|
export const BlockerNoticeRecoveryIndicators: Story = {
|
|
render: () => (
|
|
<StoryFrame
|
|
title="Blocker notice recovery indicators"
|
|
description="Blocker chips inline a recovery indicator when the blocker has an active recovery action. Plain blockers stay clean."
|
|
>
|
|
<BlockerNoticePanel />
|
|
</StoryFrame>
|
|
),
|
|
};
|
|
|
|
export const ActiveRunPanelRecoveryChips: Story = {
|
|
render: () => (
|
|
<StoryFrame
|
|
title="Active run panel recovery chips"
|
|
description="Active run cards on the dashboard expose recovery state on the linked source issue."
|
|
>
|
|
<ActiveRunPanel />
|
|
</StoryFrame>
|
|
),
|
|
};
|