forked from farhoodlabs/paperclip
d734bd43d1
## Thinking Path > - Paperclip is the control plane for autonomous AI companies, so agent work needs visible ownership, recovery, and operator controls. > - This local branch had accumulated several related control-plane reliability and operator-experience fixes across recovery actions, watchdog folding, model-profile defaults, mentions, markdown editing, plugin launchers, and small UI polish. > - The branch needed to be converted into a PR against the current `origin/master` without losing dirty work or including lockfile/workflow churn. > - The safest standalone shape is a single rollup PR because the recovery/server/UI files overlap heavily across the local commits and splitting would create avoidable conflicts. > - This pull request replays the local branch onto latest `origin/master`, preserves the uncommitted work as logical commits, and adds a Zod 4 validator compatibility fix found during verification. > - The benefit is that the May 17 local branch can be reviewed and merged as one coherent, conflict-free branch under the 100-file Greptile limit. ## What Changed - Rebased the local May 17 branch work onto current `origin/master` in a dedicated worktree. - Preserved and committed previously dirty changes for recovery retry handling, plugin/sidebar launcher polish, and `.herenow` ignores. - Added recovery-action behavior for returning source issues to `todo` when retrying source-scoped recovery. - Included the existing local recovery/liveness/watchdog fold, Codex cheap-profile, markdown/mention, duplicate-agent, and UI polish commits from the branch. - Normalized shared validator `z.record(...)` schemas to explicit string-key records for Zod 4 compatibility. - Confirmed the PR has no `pnpm-lock.yaml` or `.github/workflows/*` changes and stays below the 100-file Greptile limit. ## Verification - `pnpm install --frozen-lockfile --ignore-scripts` - `npm run install` in `node_modules/.pnpm/sqlite3@5.1.7/node_modules/sqlite3` to build the local native sqlite3 binding after installing with scripts disabled - `pnpm exec vitest run packages/shared/src/validators/issue.test.ts packages/shared/src/project-mentions.test.ts packages/adapter-utils/src/server-utils.test.ts server/src/__tests__/heartbeat-model-profile.test.ts server/src/__tests__/issue-recovery-actions.test.ts server/src/__tests__/issue-agent-mutation-ownership-routes.test.ts server/src/__tests__/heartbeat-active-run-output-watchdog.test.ts server/src/__tests__/plugin-local-folders.test.ts ui/src/components/IssueRecoveryActionCard.test.tsx ui/src/components/Sidebar.test.tsx ui/src/components/SidebarAccountMenu.test.tsx ui/src/components/IssueProperties.test.tsx ui/src/components/MarkdownEditor.test.tsx ui/src/components/MarkdownBody.test.tsx ui/src/lib/duplicate-agent-payload.test.ts ui/src/pages/Routines.test.tsx` - First pass: 13 files passed with 201 passing tests; 3 server files failed before sqlite3 native binding was built. - After rebuilding sqlite3: `server/src/__tests__/heartbeat-model-profile.test.ts`, `server/src/__tests__/issue-recovery-actions.test.ts`, and `server/src/__tests__/heartbeat-active-run-output-watchdog.test.ts` passed/loaded; embedded Postgres tests were skipped by the local host guard. - `pnpm --filter @paperclipai/shared typecheck` - `pnpm --filter @paperclipai/adapter-utils typecheck` - `pnpm --filter @paperclipai/server typecheck` - `pnpm --filter @paperclipai/ui typecheck` ## Risks - Medium risk: this is a broad rollup PR across recovery semantics, server tests, shared validators, and UI surfaces. - Some embedded Postgres tests skipped locally due the host guard, so CI should provide the stronger database-backed signal. - UI changes were covered by component tests, but no browser screenshot was captured in this PR creation pass. - This branch may overlap with existing recovery/liveness PR work; merge this PR independently or restack/close overlapping branches rather than merging duplicate implementations together. > 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-based coding agent, tool-enabled local repository and GitHub workflow, medium reasoning effort. ## 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 - [ ] 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>
1316 lines
40 KiB
TypeScript
1316 lines
40 KiB
TypeScript
// @vitest-environment jsdom
|
|
|
|
import { act } from "react";
|
|
import type { ComponentProps, ReactNode } from "react";
|
|
import { createRoot } from "react-dom/client";
|
|
import type {
|
|
ExecutionWorkspace,
|
|
IssueExecutionPolicy,
|
|
IssueExecutionState,
|
|
IssueLabel,
|
|
Project,
|
|
WorkspaceRuntimeService,
|
|
} from "@paperclipai/shared";
|
|
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
|
|
import type { Issue } from "@paperclipai/shared";
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
import { IssueProperties } from "./IssueProperties";
|
|
|
|
const mockAgentsApi = vi.hoisted(() => ({
|
|
list: vi.fn(),
|
|
adapterModels: vi.fn(),
|
|
adapterModelProfiles: vi.fn(),
|
|
}));
|
|
|
|
const mockProjectsApi = vi.hoisted(() => ({
|
|
list: vi.fn(),
|
|
}));
|
|
|
|
const mockIssuesApi = vi.hoisted(() => ({
|
|
list: vi.fn(),
|
|
listLabels: vi.fn(),
|
|
createLabel: vi.fn(),
|
|
}));
|
|
|
|
const mockAuthApi = vi.hoisted(() => ({
|
|
getSession: vi.fn(),
|
|
}));
|
|
|
|
vi.mock("../context/CompanyContext", () => ({
|
|
useCompany: () => ({
|
|
selectedCompanyId: "company-1",
|
|
}),
|
|
}));
|
|
|
|
vi.mock("../api/agents", () => ({
|
|
agentsApi: mockAgentsApi,
|
|
}));
|
|
|
|
vi.mock("../api/projects", () => ({
|
|
projectsApi: mockProjectsApi,
|
|
}));
|
|
|
|
vi.mock("../api/issues", () => ({
|
|
issuesApi: mockIssuesApi,
|
|
}));
|
|
|
|
vi.mock("../api/auth", () => ({
|
|
authApi: mockAuthApi,
|
|
}));
|
|
|
|
vi.mock("../context/ToastContext", () => ({
|
|
useToastActions: () => ({ pushToast: vi.fn() }),
|
|
}));
|
|
|
|
vi.mock("../hooks/useProjectOrder", () => ({
|
|
useProjectOrder: ({ projects }: { projects: unknown[] }) => ({
|
|
orderedProjects: projects,
|
|
}),
|
|
}));
|
|
|
|
vi.mock("../lib/recent-assignees", () => ({
|
|
getRecentAssigneeIds: () => [],
|
|
getRecentAssigneeSelectionIds: () => [],
|
|
sortAgentsByRecency: (agents: unknown[]) => agents,
|
|
trackRecentAssignee: vi.fn(),
|
|
trackRecentAssigneeUser: vi.fn(),
|
|
}));
|
|
|
|
vi.mock("../lib/assignees", () => ({
|
|
formatAssigneeUserLabel: () => "Me",
|
|
}));
|
|
|
|
vi.mock("./StatusIcon", () => ({
|
|
StatusIcon: ({ status, blockerAttention }: { status: string; blockerAttention?: Issue["blockerAttention"] }) => (
|
|
<span data-status-icon-state={blockerAttention?.state}>{status}</span>
|
|
),
|
|
}));
|
|
|
|
vi.mock("./PriorityIcon", () => ({
|
|
PriorityIcon: ({ priority }: { priority: string }) => <span>{priority}</span>,
|
|
}));
|
|
|
|
vi.mock("./Identity", () => ({
|
|
Identity: ({ name }: { name: string }) => <span>{name}</span>,
|
|
}));
|
|
|
|
vi.mock("./AgentIconPicker", () => ({
|
|
AgentIcon: () => null,
|
|
}));
|
|
|
|
vi.mock("@/lib/router", () => ({
|
|
Link: ({ children, to, ...props }: { children: ReactNode; to: string } & ComponentProps<"a">) => <a href={to} {...props}>{children}</a>,
|
|
}));
|
|
|
|
vi.mock("@/components/ui/separator", () => ({
|
|
Separator: () => <hr />,
|
|
}));
|
|
|
|
vi.mock("@/components/ui/popover", () => ({
|
|
Popover: ({ children }: { children: ReactNode }) => <div>{children}</div>,
|
|
PopoverTrigger: ({ children }: { children: ReactNode }) => <>{children}</>,
|
|
PopoverContent: ({ children }: { children: ReactNode }) => <div>{children}</div>,
|
|
}));
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
(globalThis as any).IS_REACT_ACT_ENVIRONMENT = true;
|
|
|
|
async function flush() {
|
|
await act(async () => {
|
|
await new Promise((resolve) => setTimeout(resolve, 0));
|
|
});
|
|
}
|
|
|
|
async function waitForAssertion(assertion: () => void, attempts = 20) {
|
|
let lastError: unknown;
|
|
|
|
for (let attempt = 0; attempt < attempts; attempt += 1) {
|
|
try {
|
|
assertion();
|
|
return;
|
|
} catch (error) {
|
|
lastError = error;
|
|
await flush();
|
|
}
|
|
}
|
|
|
|
throw lastError;
|
|
}
|
|
|
|
function createIssue(overrides: Partial<Issue> = {}): Issue {
|
|
return {
|
|
id: "issue-1",
|
|
companyId: "company-1",
|
|
projectId: null,
|
|
projectWorkspaceId: null,
|
|
goalId: null,
|
|
parentId: null,
|
|
title: "Parent issue",
|
|
description: null,
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
checkoutRunId: null,
|
|
executionRunId: null,
|
|
executionAgentNameKey: null,
|
|
executionLockedAt: null,
|
|
createdByAgentId: null,
|
|
createdByUserId: "user-1",
|
|
issueNumber: 1,
|
|
identifier: "PAP-1",
|
|
requestDepth: 0,
|
|
billingCode: null,
|
|
assigneeAdapterOverrides: null,
|
|
executionWorkspaceId: null,
|
|
executionWorkspacePreference: null,
|
|
executionWorkspaceSettings: null,
|
|
startedAt: null,
|
|
completedAt: null,
|
|
cancelledAt: null,
|
|
hiddenAt: null,
|
|
labels: [],
|
|
labelIds: [],
|
|
blockedBy: [],
|
|
blocks: [],
|
|
createdAt: new Date("2026-04-06T12:00:00.000Z"),
|
|
updatedAt: new Date("2026-04-06T12:05:00.000Z"),
|
|
...overrides,
|
|
workMode: overrides.workMode ?? "standard",
|
|
};
|
|
}
|
|
|
|
function createLabel(overrides: Partial<IssueLabel> = {}): IssueLabel {
|
|
return {
|
|
id: "label-1",
|
|
companyId: "company-1",
|
|
name: "Bug",
|
|
color: "#ef4444",
|
|
createdAt: new Date("2026-04-06T12:00:00.000Z"),
|
|
updatedAt: new Date("2026-04-06T12:00:00.000Z"),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function createRuntimeService(overrides: Partial<WorkspaceRuntimeService> = {}): WorkspaceRuntimeService {
|
|
return {
|
|
id: "service-1",
|
|
companyId: "company-1",
|
|
projectId: "project-1",
|
|
projectWorkspaceId: "workspace-main",
|
|
executionWorkspaceId: "workspace-1",
|
|
issueId: "issue-1",
|
|
scopeType: "execution_workspace",
|
|
scopeId: "workspace-1",
|
|
serviceName: "web",
|
|
status: "running",
|
|
lifecycle: "shared",
|
|
reuseKey: null,
|
|
command: "pnpm dev",
|
|
cwd: "/tmp/paperclip",
|
|
port: 62475,
|
|
url: "http://127.0.0.1:62475",
|
|
provider: "local_process",
|
|
providerRef: null,
|
|
ownerAgentId: null,
|
|
startedByRunId: null,
|
|
lastUsedAt: new Date("2026-04-06T12:03:00.000Z"),
|
|
startedAt: new Date("2026-04-06T12:02:00.000Z"),
|
|
stoppedAt: null,
|
|
stopPolicy: null,
|
|
healthStatus: "healthy",
|
|
createdAt: new Date("2026-04-06T12:02:00.000Z"),
|
|
updatedAt: new Date("2026-04-06T12:03:00.000Z"),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function createExecutionWorkspace(overrides: Partial<ExecutionWorkspace> = {}): ExecutionWorkspace {
|
|
return {
|
|
id: "workspace-1",
|
|
companyId: "company-1",
|
|
projectId: "project-1",
|
|
projectWorkspaceId: "workspace-main",
|
|
sourceIssueId: "issue-1",
|
|
mode: "isolated_workspace",
|
|
strategyType: "git_worktree",
|
|
name: "PAP-1 workspace",
|
|
status: "active",
|
|
cwd: "/tmp/paperclip/PAP-1",
|
|
repoUrl: null,
|
|
baseRef: "master",
|
|
branchName: "pap-1-workspace",
|
|
providerType: "git_worktree",
|
|
providerRef: "/tmp/paperclip/PAP-1",
|
|
derivedFromExecutionWorkspaceId: null,
|
|
lastUsedAt: new Date("2026-04-06T12:04:00.000Z"),
|
|
openedAt: new Date("2026-04-06T12:01:00.000Z"),
|
|
closedAt: null,
|
|
cleanupEligibleAt: null,
|
|
cleanupReason: null,
|
|
config: null,
|
|
metadata: null,
|
|
runtimeServices: [createRuntimeService()],
|
|
createdAt: new Date("2026-04-06T12:01:00.000Z"),
|
|
updatedAt: new Date("2026-04-06T12:04:00.000Z"),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function createProject(overrides: Partial<Project> = {}): Project {
|
|
const primaryWorkspace = {
|
|
id: "workspace-main",
|
|
companyId: "company-1",
|
|
projectId: "project-1",
|
|
name: "Main",
|
|
sourceType: "local_path" as const,
|
|
cwd: "/tmp/paperclip",
|
|
repoUrl: null,
|
|
repoRef: null,
|
|
defaultRef: "master",
|
|
visibility: "default" as const,
|
|
setupCommand: null,
|
|
cleanupCommand: null,
|
|
remoteProvider: null,
|
|
remoteWorkspaceRef: null,
|
|
sharedWorkspaceKey: null,
|
|
metadata: null,
|
|
runtimeConfig: null,
|
|
isPrimary: true,
|
|
runtimeServices: [],
|
|
createdAt: new Date("2026-04-06T12:00:00.000Z"),
|
|
updatedAt: new Date("2026-04-06T12:00:00.000Z"),
|
|
};
|
|
return {
|
|
id: "project-1",
|
|
companyId: "company-1",
|
|
urlKey: "project-1",
|
|
goalId: null,
|
|
goalIds: [],
|
|
goals: [],
|
|
name: "Project",
|
|
description: null,
|
|
status: "in_progress",
|
|
leadAgentId: null,
|
|
targetDate: null,
|
|
color: "#6366f1",
|
|
env: null,
|
|
pauseReason: null,
|
|
pausedAt: null,
|
|
executionWorkspacePolicy: null,
|
|
codebase: {
|
|
workspaceId: "workspace-main",
|
|
repoUrl: null,
|
|
repoRef: null,
|
|
defaultRef: "master",
|
|
repoName: null,
|
|
localFolder: "/tmp/paperclip",
|
|
managedFolder: "/tmp/paperclip",
|
|
effectiveLocalFolder: "/tmp/paperclip",
|
|
origin: "local_folder",
|
|
},
|
|
workspaces: [primaryWorkspace],
|
|
primaryWorkspace,
|
|
archivedAt: null,
|
|
createdAt: new Date("2026-04-06T12:00:00.000Z"),
|
|
updatedAt: new Date("2026-04-06T12:00:00.000Z"),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function createExecutionPolicy(overrides: Partial<IssueExecutionPolicy> = {}): IssueExecutionPolicy {
|
|
return {
|
|
mode: "normal",
|
|
commentRequired: true,
|
|
stages: [],
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function createExecutionState(overrides: Partial<IssueExecutionState> = {}): IssueExecutionState {
|
|
return {
|
|
status: "changes_requested",
|
|
currentStageId: "stage-1",
|
|
currentStageIndex: 0,
|
|
currentStageType: "review",
|
|
currentParticipant: { type: "agent", agentId: "agent-1", userId: null },
|
|
returnAssignee: { type: "agent", agentId: "agent-2", userId: null },
|
|
reviewRequest: null,
|
|
completedStageIds: [],
|
|
lastDecisionId: null,
|
|
lastDecisionOutcome: "changes_requested",
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function renderProperties(container: HTMLDivElement, props: ComponentProps<typeof IssueProperties>) {
|
|
const queryClient = new QueryClient({
|
|
defaultOptions: {
|
|
queries: { retry: false },
|
|
},
|
|
});
|
|
const root = createRoot(container);
|
|
act(() => {
|
|
root.render(
|
|
<QueryClientProvider client={queryClient}>
|
|
<IssueProperties {...props} />
|
|
</QueryClientProvider>,
|
|
);
|
|
});
|
|
return root;
|
|
}
|
|
|
|
describe("IssueProperties", () => {
|
|
let container: HTMLDivElement;
|
|
|
|
beforeEach(() => {
|
|
container = document.createElement("div");
|
|
document.body.appendChild(container);
|
|
mockAgentsApi.list.mockResolvedValue([]);
|
|
mockAgentsApi.adapterModels.mockResolvedValue([]);
|
|
mockAgentsApi.adapterModelProfiles.mockResolvedValue([]);
|
|
mockProjectsApi.list.mockResolvedValue([]);
|
|
mockIssuesApi.list.mockResolvedValue([]);
|
|
mockIssuesApi.listLabels.mockResolvedValue([]);
|
|
mockIssuesApi.createLabel.mockResolvedValue(createLabel({
|
|
id: "label-new",
|
|
name: "New label",
|
|
color: "#6366f1",
|
|
}));
|
|
mockAuthApi.getSession.mockResolvedValue({ user: { id: "user-1" } });
|
|
});
|
|
|
|
afterEach(() => {
|
|
document.body.innerHTML = "";
|
|
});
|
|
|
|
it("always exposes the add sub-issue action", async () => {
|
|
const onAddSubIssue = vi.fn();
|
|
const root = renderProperties(container, {
|
|
issue: createIssue(),
|
|
childIssues: [],
|
|
onAddSubIssue,
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).toContain("Sub-issues");
|
|
expect(container.textContent).toContain("Add sub-issue");
|
|
|
|
const addButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("Add sub-issue"));
|
|
expect(addButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
addButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onAddSubIssue).toHaveBeenCalledTimes(1);
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("passes blocker attention to the sidebar status icon", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
status: "blocked",
|
|
blockerAttention: {
|
|
state: "covered",
|
|
reason: "active_child",
|
|
unresolvedBlockerCount: 1,
|
|
coveredBlockerCount: 1,
|
|
stalledBlockerCount: 0,
|
|
attentionBlockerCount: 0,
|
|
sampleBlockerIdentifier: "PAP-2",
|
|
sampleStalledBlockerIdentifier: null,
|
|
},
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.querySelector('[data-status-icon-state="covered"]')?.textContent).toBe("blocked");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("renders blocked-by issues as direct chips and edits them from an add action", async () => {
|
|
const onUpdate = vi.fn();
|
|
mockIssuesApi.list.mockResolvedValue([
|
|
createIssue({ id: "issue-3", identifier: "PAP-3", title: "New blocker", status: "todo" }),
|
|
]);
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
blockedBy: [
|
|
{
|
|
id: "issue-2",
|
|
identifier: "PAP-2",
|
|
title: "Existing blocker",
|
|
status: "in_progress",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
],
|
|
}),
|
|
childIssues: [],
|
|
onUpdate,
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
const blockerLink = container.querySelector('a[href="/issues/PAP-2"]');
|
|
expect(blockerLink).not.toBeNull();
|
|
expect(blockerLink?.textContent).toContain("PAP-2");
|
|
expect(blockerLink?.closest("button")).toBeNull();
|
|
expect(container.textContent).toContain("Add blocker");
|
|
expect(container.querySelector('input[placeholder="Search issues..."]')).toBeNull();
|
|
|
|
const addButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("Add blocker"));
|
|
expect(addButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
addButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
expect(container.querySelector('input[placeholder="Search issues..."]')).not.toBeNull();
|
|
|
|
const candidateButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("PAP-3 New blocker"));
|
|
expect(candidateButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
candidateButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({ blockedByIssueIds: ["issue-2", "issue-3"] });
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("searches all company issues when adding a blocker", async () => {
|
|
const onUpdate = vi.fn();
|
|
const loadedIssue = createIssue({ id: "issue-3", identifier: "PAP-3", title: "Loaded issue", status: "todo" });
|
|
const remoteIssue = createIssue({ id: "issue-99", identifier: "PAP-99", title: "Remote blocker", status: "in_progress" });
|
|
mockIssuesApi.list.mockImplementation((_companyId: string, filters?: { q?: string; limit?: number }) => {
|
|
if (filters?.q === "remote") return Promise.resolve([remoteIssue]);
|
|
return Promise.resolve([loadedIssue]);
|
|
});
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue(),
|
|
childIssues: [],
|
|
onUpdate,
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
const addButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("Add blocker"));
|
|
expect(addButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
addButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
const searchInput = container.querySelector('input[aria-label="Search issues to add as blockers"]') as HTMLInputElement | null;
|
|
expect(searchInput).not.toBeNull();
|
|
|
|
await act(async () => {
|
|
const nativeSetter = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, "value")?.set;
|
|
nativeSetter?.call(searchInput, "remote");
|
|
searchInput!.dispatchEvent(new Event("input", { bubbles: true }));
|
|
});
|
|
|
|
await waitForAssertion(() => {
|
|
expect(mockIssuesApi.list).toHaveBeenCalledWith("company-1", { q: "remote", limit: 50 });
|
|
expect(container.textContent).toContain("PAP-99 Remote blocker");
|
|
expect(container.textContent).not.toContain("PAP-3 Loaded issue");
|
|
});
|
|
|
|
const candidateButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("PAP-99 Remote blocker"));
|
|
expect(candidateButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
candidateButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({ blockedByIssueIds: ["issue-99"] });
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("removes a blocked-by issue from the chip remove action after confirmation", async () => {
|
|
const onUpdate = vi.fn();
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
blockedBy: [
|
|
{
|
|
id: "issue-2",
|
|
identifier: "PAP-2",
|
|
title: "Existing blocker",
|
|
status: "in_progress",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
{
|
|
id: "issue-4",
|
|
identifier: "PAP-4",
|
|
title: "Keep blocker",
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
],
|
|
}),
|
|
childIssues: [],
|
|
onUpdate,
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
const removeButton = container.querySelector('button[aria-label="Remove PAP-2 as blocker"]');
|
|
expect(removeButton).not.toBeNull();
|
|
|
|
await act(async () => {
|
|
removeButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
expect(document.body.textContent).toContain("Remove PAP-2: Existing blocker as a blocker for this issue.");
|
|
const confirmButton = Array.from(document.body.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("Remove blocker"));
|
|
expect(confirmButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
confirmButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({ blockedByIssueIds: ["issue-4"] });
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows a green service link above the workspace row for a live non-main workspace", async () => {
|
|
mockProjectsApi.list.mockResolvedValue([createProject()]);
|
|
const serviceUrl = "http://127.0.0.1:62475";
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
projectId: "project-1",
|
|
projectWorkspaceId: "workspace-main",
|
|
executionWorkspaceId: "workspace-1",
|
|
currentExecutionWorkspace: createExecutionWorkspace({
|
|
mode: "isolated_workspace",
|
|
runtimeServices: [createRuntimeService({ url: serviceUrl, status: "running" })],
|
|
}),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
const serviceLink = container.querySelector(`a[href="${serviceUrl}"]`);
|
|
expect(serviceLink).not.toBeNull();
|
|
expect(serviceLink?.getAttribute("target")).toBe("_blank");
|
|
expect(serviceLink?.className).toContain("text-emerald");
|
|
expect((container.textContent ?? "").indexOf("Service")).toBeLessThan(
|
|
(container.textContent ?? "").indexOf("Workspace"),
|
|
);
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows full date and time for issue metadata timestamps", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
createdAt: new Date(2026, 3, 6, 12, 34),
|
|
startedAt: new Date(2026, 3, 6, 12, 35),
|
|
completedAt: new Date(2026, 3, 6, 12, 36),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).toMatch(/CreatedApr 6, 2026, \d{1,2}:34 (AM|PM)/);
|
|
expect(container.textContent).toMatch(/StartedApr 6, 2026, \d{1,2}:35 (AM|PM)/);
|
|
expect(container.textContent).toMatch(/CompletedApr 6, 2026, \d{1,2}:36 (AM|PM)/);
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows only the workspace detail link for non-default workspaces", async () => {
|
|
mockProjectsApi.list.mockResolvedValue([createProject()]);
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
projectId: "project-1",
|
|
projectWorkspaceId: "workspace-main",
|
|
executionWorkspaceId: "workspace-1",
|
|
currentExecutionWorkspace: createExecutionWorkspace({
|
|
mode: "isolated_workspace",
|
|
}),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
await flush();
|
|
|
|
const workspaceLink = Array.from(container.querySelectorAll("a")).find(
|
|
(link) => link.textContent?.trim() === "View workspace",
|
|
);
|
|
expect(container.textContent).not.toContain("View workspace tasks");
|
|
expect(workspaceLink).not.toBeUndefined();
|
|
expect(workspaceLink?.getAttribute("href")).toBe("/execution-workspaces/workspace-1");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("does not show a service link for the main shared workspace", async () => {
|
|
mockProjectsApi.list.mockResolvedValue([createProject()]);
|
|
const serviceUrl = "http://127.0.0.1:62475";
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
projectId: "project-1",
|
|
projectWorkspaceId: "workspace-main",
|
|
executionWorkspaceId: "workspace-1",
|
|
currentExecutionWorkspace: createExecutionWorkspace({
|
|
mode: "shared_workspace",
|
|
projectWorkspaceId: "workspace-main",
|
|
runtimeServices: [createRuntimeService({ url: serviceUrl, status: "running" })],
|
|
}),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.querySelector(`a[href="${serviceUrl}"]`)).toBeNull();
|
|
expect(container.textContent).not.toContain("View workspace tasks");
|
|
expect(Array.from(container.querySelectorAll("a")).some(
|
|
(link) => link.textContent?.trim() === "View workspace",
|
|
)).toBe(false);
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows related task references below sub-issues", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
relatedWork: {
|
|
outbound: [
|
|
{
|
|
issue: {
|
|
id: "issue-22",
|
|
identifier: "PAP-22",
|
|
title: "Related task",
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
mentionCount: 1,
|
|
sources: [{ kind: "description", sourceRecordId: null, label: "description", matchedText: "PAP-22" }],
|
|
},
|
|
],
|
|
inbound: [],
|
|
},
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).not.toContain("Task ids");
|
|
expect(container.textContent).toContain("Related Tasks");
|
|
expect(container.textContent).toContain("PAP-22");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("hides related task references already covered by blockers, blocking, and sub-issues", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
blockedBy: [
|
|
{
|
|
id: "issue-22",
|
|
identifier: "PAP-22",
|
|
title: "Blocker",
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
],
|
|
blocks: [
|
|
{
|
|
id: "issue-33",
|
|
identifier: "PAP-33",
|
|
title: "Blocked issue",
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
],
|
|
relatedWork: {
|
|
outbound: [
|
|
{
|
|
issue: {
|
|
id: "issue-22",
|
|
identifier: "PAP-22",
|
|
title: "Blocker",
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
mentionCount: 1,
|
|
sources: [{ kind: "description", sourceRecordId: null, label: "description", matchedText: "PAP-22" }],
|
|
},
|
|
{
|
|
issue: {
|
|
id: "issue-33",
|
|
identifier: "PAP-33",
|
|
title: "Blocked issue",
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
mentionCount: 1,
|
|
sources: [{ kind: "description", sourceRecordId: null, label: "description", matchedText: "PAP-33" }],
|
|
},
|
|
{
|
|
issue: {
|
|
id: "child-44",
|
|
identifier: "PAP-44",
|
|
title: "Child issue",
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
},
|
|
mentionCount: 1,
|
|
sources: [{ kind: "description", sourceRecordId: null, label: "description", matchedText: "PAP-44" }],
|
|
},
|
|
],
|
|
inbound: [],
|
|
},
|
|
}),
|
|
childIssues: [
|
|
createIssue({
|
|
id: "child-44",
|
|
identifier: "PAP-44",
|
|
title: "Child issue",
|
|
}),
|
|
],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).not.toContain("Related Tasks");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows an add-label button when labels already exist and opens the picker", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
labels: [{ id: "label-1", companyId: "company-1", name: "Bug", color: "#ef4444", createdAt: new Date("2026-04-06T12:00:00.000Z"), updatedAt: new Date("2026-04-06T12:00:00.000Z") }],
|
|
labelIds: ["label-1"],
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
const addLabelButton = container.querySelector('button[aria-label="Add label"]');
|
|
expect(addLabelButton).not.toBeNull();
|
|
expect(container.querySelector('input[placeholder="Search labels..."]')).toBeNull();
|
|
|
|
await act(async () => {
|
|
addLabelButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
expect(container.querySelector('input[placeholder="Search labels..."]')).not.toBeNull();
|
|
expect(container.querySelector('button[title="Delete Bug"]')).toBeNull();
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows selected labels from labelIds even before the issue labels relation refreshes", async () => {
|
|
mockIssuesApi.listLabels.mockResolvedValue([createLabel()]);
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
labels: [],
|
|
labelIds: ["label-1"],
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
await flush();
|
|
|
|
expect(container.textContent).toContain("Bug");
|
|
expect(container.textContent).not.toContain("No labels");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("hides model options when the issue uses the assignee default", async () => {
|
|
mockAgentsApi.list.mockResolvedValue([
|
|
{
|
|
id: "agent-1",
|
|
name: "Senior Product Engineer",
|
|
role: "engineer",
|
|
title: null,
|
|
status: "active",
|
|
adapterType: "codex_local",
|
|
icon: null,
|
|
},
|
|
]);
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
assigneeAgentId: "agent-1",
|
|
assigneeAdapterOverrides: null,
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).not.toContain("Model lane");
|
|
expect(container.textContent).not.toContain("Codex options");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("edits existing custom assignee model options from the properties pane", async () => {
|
|
const onUpdate = vi.fn();
|
|
mockAgentsApi.list.mockResolvedValue([
|
|
{
|
|
id: "agent-1",
|
|
name: "Senior Product Engineer",
|
|
role: "engineer",
|
|
title: null,
|
|
status: "active",
|
|
adapterType: "codex_local",
|
|
icon: null,
|
|
},
|
|
]);
|
|
mockAgentsApi.adapterModels.mockResolvedValue([
|
|
{ id: "gpt-5.5", label: "GPT-5.5" },
|
|
{ id: "gpt-5.4", label: "GPT-5.4" },
|
|
]);
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
assigneeAgentId: "agent-1",
|
|
assigneeAdapterOverrides: {
|
|
adapterConfig: {
|
|
model: "gpt-5.4",
|
|
modelReasoningEffort: "high",
|
|
},
|
|
},
|
|
}),
|
|
childIssues: [],
|
|
onUpdate,
|
|
});
|
|
await flush();
|
|
await flush();
|
|
|
|
expect(container.textContent).toContain("Custom · gpt-5.4 · high");
|
|
expect(container.textContent).toContain("Model lane");
|
|
|
|
const modelButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("GPT-5.5"));
|
|
expect(modelButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
modelButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({
|
|
assigneeAdapterOverrides: {
|
|
adapterConfig: {
|
|
model: "gpt-5.5",
|
|
modelReasoningEffort: "high",
|
|
},
|
|
},
|
|
});
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("clears existing assignee adapter overrides from the properties pane", async () => {
|
|
const onUpdate = vi.fn();
|
|
mockAgentsApi.list.mockResolvedValue([
|
|
{
|
|
id: "agent-1",
|
|
name: "Senior Product Engineer",
|
|
role: "engineer",
|
|
title: null,
|
|
status: "active",
|
|
adapterType: "codex_local",
|
|
icon: null,
|
|
},
|
|
]);
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
assigneeAgentId: "agent-1",
|
|
assigneeAdapterOverrides: {
|
|
adapterConfig: {
|
|
model: "gpt-5.4",
|
|
},
|
|
},
|
|
}),
|
|
childIssues: [],
|
|
onUpdate,
|
|
});
|
|
await flush();
|
|
|
|
const clearButton = container.querySelector('button[aria-label="Clear adapter options"]');
|
|
expect(clearButton).not.toBeNull();
|
|
|
|
await act(async () => {
|
|
clearButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({ assigneeAdapterOverrides: null });
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows a checkmark on selected labels in the picker", async () => {
|
|
mockIssuesApi.listLabels.mockResolvedValue([
|
|
createLabel(),
|
|
createLabel({ id: "label-2", name: "Feature", color: "#22c55e" }),
|
|
]);
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
labels: [createLabel()],
|
|
labelIds: ["label-1"],
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
const addLabelButton = container.querySelector('button[aria-label="Add label"]');
|
|
expect(addLabelButton).not.toBeNull();
|
|
await act(async () => {
|
|
addLabelButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
const labelButtons = Array.from(container.querySelectorAll("button"))
|
|
.filter((button) => button.textContent?.includes("Bug") || button.textContent?.includes("Feature"));
|
|
const bugButton = labelButtons.find((button) => button.textContent?.includes("Bug") && button.querySelector("svg"));
|
|
const featureButton = labelButtons.find((button) => button.textContent?.includes("Feature"));
|
|
expect(bugButton).not.toBeUndefined();
|
|
expect(featureButton?.querySelector("svg")).toBeNull();
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("allows setting and clearing a parent issue from the properties pane", async () => {
|
|
const onUpdate = vi.fn();
|
|
mockIssuesApi.list.mockResolvedValue([
|
|
createIssue({ id: "issue-2", identifier: "PAP-2", title: "Candidate parent", status: "in_progress" }),
|
|
]);
|
|
|
|
const root = renderProperties(container, {
|
|
issue: createIssue(),
|
|
childIssues: [],
|
|
onUpdate,
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
const parentTrigger = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("No parent"));
|
|
expect(parentTrigger).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
parentTrigger!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
const candidateButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("PAP-2 Candidate parent"));
|
|
expect(candidateButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
candidateButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({ parentId: "issue-2" });
|
|
|
|
onUpdate.mockClear();
|
|
const rerenderedIssue = createIssue({
|
|
parentId: "issue-2",
|
|
ancestors: [
|
|
{
|
|
id: "issue-2",
|
|
identifier: "PAP-2",
|
|
title: "Candidate parent",
|
|
description: null,
|
|
status: "in_progress",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
projectId: null,
|
|
goalId: null,
|
|
project: null,
|
|
goal: null,
|
|
},
|
|
],
|
|
});
|
|
|
|
act(() => root.unmount());
|
|
|
|
const rerenderedRoot = renderProperties(container, {
|
|
issue: rerenderedIssue,
|
|
childIssues: [],
|
|
onUpdate,
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
const selectedParentTrigger = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("PAP-2 Candidate parent"));
|
|
expect(selectedParentTrigger).not.toBeUndefined();
|
|
const parentLink = container.querySelector('a[href="/issues/PAP-2"]');
|
|
expect(parentLink).not.toBeNull();
|
|
expect(selectedParentTrigger!.contains(parentLink)).toBe(false);
|
|
|
|
await act(async () => {
|
|
selectedParentTrigger!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
const clearParentButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("No parent"));
|
|
expect(clearParentButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
clearParentButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({ parentId: null });
|
|
|
|
act(() => rerenderedRoot.unmount());
|
|
});
|
|
it("shows a run review action after reviewers are configured and starts execution explicitly when clicked", async () => {
|
|
const onUpdate = vi.fn();
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
executionPolicy: createExecutionPolicy({
|
|
stages: [
|
|
{
|
|
id: "review-stage",
|
|
type: "review",
|
|
approvalsNeeded: 1,
|
|
participants: [{ id: "participant-1", type: "agent", agentId: "agent-1", userId: null }],
|
|
},
|
|
],
|
|
}),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate,
|
|
});
|
|
await flush();
|
|
|
|
const runReviewButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("Run review now"));
|
|
expect(runReviewButton).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
runReviewButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({ status: "in_review" });
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("shows a run approval action when approval is the next runnable stage", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
executionPolicy: createExecutionPolicy({
|
|
stages: [
|
|
{
|
|
id: "approval-stage",
|
|
type: "approval",
|
|
approvalsNeeded: 1,
|
|
participants: [{ id: "participant-2", type: "user", agentId: null, userId: "user-1" }],
|
|
},
|
|
],
|
|
}),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).toContain("Run approval now");
|
|
expect(container.textContent).not.toContain("Run review now");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("keeps the run review action available after changes are requested", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
status: "in_progress",
|
|
executionPolicy: createExecutionPolicy({
|
|
stages: [
|
|
{
|
|
id: "review-stage",
|
|
type: "review",
|
|
approvalsNeeded: 1,
|
|
participants: [{ id: "participant-1", type: "agent", agentId: "agent-1", userId: null }],
|
|
},
|
|
],
|
|
}),
|
|
executionState: createExecutionState(),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).toContain("Run review now");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("hides the run action while an execution stage is already pending", async () => {
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
status: "in_review",
|
|
executionPolicy: createExecutionPolicy({
|
|
stages: [
|
|
{
|
|
id: "review-stage",
|
|
type: "review",
|
|
approvalsNeeded: 1,
|
|
participants: [{ id: "participant-1", type: "agent", agentId: "agent-1", userId: null }],
|
|
},
|
|
],
|
|
}),
|
|
executionState: createExecutionState({
|
|
status: "pending",
|
|
currentStageType: "review",
|
|
lastDecisionOutcome: null,
|
|
}),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate: vi.fn(),
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).not.toContain("Run review now");
|
|
expect(container.textContent).not.toContain("Run approval now");
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
|
|
it("renders monitor controls and clears an existing monitor", async () => {
|
|
const onUpdate = vi.fn();
|
|
const root = renderProperties(container, {
|
|
issue: createIssue({
|
|
status: "in_progress",
|
|
assigneeAgentId: "agent-1",
|
|
executionPolicy: createExecutionPolicy({
|
|
monitor: {
|
|
nextCheckAt: "2026-04-11T12:30:00.000Z",
|
|
notes: "Check deployment",
|
|
scheduledBy: "board",
|
|
},
|
|
}),
|
|
executionState: createExecutionState({
|
|
status: "idle",
|
|
currentStageId: null,
|
|
currentStageIndex: null,
|
|
currentStageType: null,
|
|
currentParticipant: null,
|
|
returnAssignee: null,
|
|
lastDecisionOutcome: null,
|
|
monitor: {
|
|
status: "scheduled",
|
|
nextCheckAt: "2026-04-11T12:30:00.000Z",
|
|
lastTriggeredAt: null,
|
|
attemptCount: 0,
|
|
notes: "Check deployment",
|
|
scheduledBy: "board",
|
|
clearedAt: null,
|
|
clearReason: null,
|
|
},
|
|
}),
|
|
}),
|
|
childIssues: [],
|
|
onUpdate,
|
|
inline: true,
|
|
});
|
|
await flush();
|
|
|
|
expect(container.textContent).toContain("Monitor");
|
|
expect(container.textContent).toContain("Next check");
|
|
expect(container.querySelector('input[type="datetime-local"]')).toBeNull();
|
|
expect(container.querySelector('input[placeholder="What should the agent re-check?"]')).toBeNull();
|
|
|
|
const monitorTrigger = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("Next check"));
|
|
expect(monitorTrigger).not.toBeUndefined();
|
|
|
|
await act(async () => {
|
|
monitorTrigger!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
await flush();
|
|
|
|
const inputs = Array.from(container.querySelectorAll("input"));
|
|
const datetimeInput = inputs.find((input) => input.getAttribute("type") === "datetime-local");
|
|
const textInput = inputs.find((input) => input.getAttribute("placeholder") === "What should the agent re-check?");
|
|
const clearButton = Array.from(container.querySelectorAll("button"))
|
|
.find((button) => button.textContent?.includes("Clear"));
|
|
|
|
expect(datetimeInput).toBeTruthy();
|
|
expect(textInput).toBeTruthy();
|
|
expect(clearButton).toBeTruthy();
|
|
expect(datetimeInput!.value).toBeTruthy();
|
|
expect(textInput!.value).toBe("Check deployment");
|
|
|
|
act(() => {
|
|
clearButton!.dispatchEvent(new MouseEvent("click", { bubbles: true }));
|
|
});
|
|
|
|
expect(onUpdate).toHaveBeenCalledWith({
|
|
executionPolicy: {
|
|
mode: "normal",
|
|
commentRequired: true,
|
|
stages: [],
|
|
},
|
|
});
|
|
|
|
act(() => root.unmount());
|
|
});
|
|
});
|