forked from farhoodlabs/paperclip
772fc92619
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies > - Issue operators need clear controls for execution settings, model overrides, and recovery retries > - Existing issue properties hid useful adapter override state and did not expose a board-triggered retry for scheduled heartbeat recovery > - Scheduled retries also need to respect the same safety gates as normal execution instead of bypassing budget, review, pause, dependency, or terminal-state checks > - This pull request adds the issue property controls and retry-now surfaces together because they share the issue details/properties UI > - The benefit is that operators can inspect and adjust issue execution settings and safely trigger pending scheduled recovery without hidden control-plane behavior ## What Changed - Adds editable issue assignee model override controls in `IssueProperties`, with focused coverage. - Removes the stale workspace tasks link from issue properties. - Adds a scheduled retry `retry-now` backend path and shared response types. - Adds main-pane and properties-pane scheduled retry UI, backed by a shared `useRetryNowMutation` hook. - Adds suppression coverage for budget hard stops, review participant changes, subtree pause holds, unresolved blockers, terminal issues, and company scoping. - Updates the `IssueProperties` test harness with toast actions required by the retry-now hook. ## Verification - `pnpm exec vitest run ui/src/components/IssueProperties.test.tsx ui/src/components/IssueScheduledRetryCard.test.tsx` — 31 passed. - `pnpm exec vitest run server/src/__tests__/issue-scheduled-retry-routes.test.ts` — exited 0, but this host skipped the embedded Postgres route tests with: `Postgres init script exited with code null. Please check the logs for extra info. The data directory might already exist.` - Pairwise merge check against the assigned-backlog PR branch completed without conflicts via `git merge --no-commit --no-ff` in a temporary worktree. ### Visual verification screenshots Storybook story: `Product/Issue Scheduled retry surfaces / ScheduledRetrySurfaces`.   ## Risks - Medium: this touches issue execution/retry behavior, so CI should run the embedded Postgres route tests on a host that can initialize Postgres. - Low-to-medium UI risk around duplicated retry-now entry points; both surfaces share one mutation hook to keep behavior consistent. > 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 coding agent, GPT-5 model family (`gpt-5`), tool-enabled Paperclip heartbeat environment. Context window and internal reasoning mode are 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>
1246 lines
37 KiB
TypeScript
1246 lines
37 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));
|
|
});
|
|
}
|
|
|
|
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("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());
|
|
});
|
|
});
|