forked from farhoodlabs/paperclip
5f45712846
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies > - The board depends on issue, inbox, cost, and company-skill surfaces to stay accurate and fast while agents are actively working > - The PAP-1497 follow-up branch exposed a few rough edges in those surfaces: stale active-run state on completed issues, missing creator filters, oversized issue payload scans, and placeholder issue-route parsing > - Those gaps make the control plane harder to trust because operators can see misleading run state, miss the right subset of work, or pay extra query/render cost on large issue records > - This pull request tightens those follow-ups across server and UI code, and adds regression coverage for the affected paths > - The benefit is a more reliable issue workflow, safer high-volume cost aggregation, and clearer board/operator navigation ## What Changed - Added the `v2026.415.0` release changelog entry. - Fixed stale issue-run presentation after completion and reused the shared issue-path parser so literal route placeholders no longer become issue links. - Added creator filters to the Issues page and Inbox, including persisted filter-state normalization and regression coverage. - Bounded issue detail/list project-mention scans and trimmed large issue-list payload fields to keep issue reads lighter. - Hardened company-skill list projection and cost/finance aggregation so large markdown blobs and large summed values do not leak into list responses or overflow 32-bit casts. - Added targeted server/UI regression tests for company skills, costs/finance, issue mention scanning, creator filters, inbox normalization, and issue reference parsing. ## Verification - `pnpm exec vitest run server/src/__tests__/company-skills-service.test.ts server/src/__tests__/costs-service.test.ts server/src/__tests__/issues-goal-context-routes.test.ts server/src/__tests__/issues-service.test.ts ui/src/lib/inbox.test.ts ui/src/lib/issue-filters.test.ts ui/src/lib/issue-reference.test.ts` - `gh pr checks 3779` Current pass set on the PR head: `policy`, `verify`, `e2e`, `security/snyk (cryppadotta)`, `Greptile Review` ## Risks - Creator filter options are derived from the currently loaded issue/agent data, so very sparse result sets may not surface every historical creator until they appear in the active dataset. - Cost/finance aggregate casts now use `double precision`; that removes the current overflow risk, but future schema changes should keep large-value aggregation behavior under review. - Issue detail mention scanning now skips comment-body scans on the detail route, so any consumer that relied on comment-only project mentions there would need to fetch them separately. ## Model Used - OpenAI Codex, GPT-5-based coding agent with terminal tool use and local code execution in the Paperclip workspace. Exact internal model ID/context-window exposure is not surfaced in this session. ## 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 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>
1159 lines
37 KiB
TypeScript
1159 lines
37 KiB
TypeScript
// @vitest-environment node
|
|
|
|
import { beforeEach, describe, expect, it } from "vitest";
|
|
import type {
|
|
Approval,
|
|
DashboardSummary,
|
|
ExecutionWorkspace,
|
|
HeartbeatRun,
|
|
Issue,
|
|
JoinRequest,
|
|
ProjectWorkspace,
|
|
} from "@paperclipai/shared";
|
|
import {
|
|
DEFAULT_INBOX_ISSUE_COLUMNS,
|
|
buildGroupedInboxSections,
|
|
buildInboxKeyboardNavEntries,
|
|
buildInboxDismissedAtByKey,
|
|
computeInboxBadgeData,
|
|
filterInboxIssues,
|
|
getArchivedInboxSearchIssues,
|
|
getAvailableInboxIssueColumns,
|
|
getInboxWorkItemKey,
|
|
getApprovalsForTab,
|
|
getInboxWorkItems,
|
|
getInboxKeyboardSelectionIndex,
|
|
getInboxSearchSupplementIssues,
|
|
getRecentTouchedIssues,
|
|
getUnreadTouchedIssues,
|
|
groupInboxWorkItems,
|
|
isInboxEntityDismissed,
|
|
isMineInboxTab,
|
|
loadInboxFilterPreferences,
|
|
loadInboxIssueColumns,
|
|
loadInboxWorkItemGroupBy,
|
|
loadCollapsedInboxGroupKeys,
|
|
loadLastInboxTab,
|
|
matchesInboxIssueSearch,
|
|
normalizeInboxIssueColumns,
|
|
RECENT_ISSUES_LIMIT,
|
|
resolveInboxNestingEnabled,
|
|
resolveIssueWorkspaceName,
|
|
resolveIssueWorkspaceGroup,
|
|
resolveInboxSelectionIndex,
|
|
saveInboxFilterPreferences,
|
|
saveCollapsedInboxGroupKeys,
|
|
saveInboxIssueColumns,
|
|
saveInboxWorkItemGroupBy,
|
|
saveLastInboxTab,
|
|
shouldResetInboxWorkspaceGrouping,
|
|
shouldShowInboxSection,
|
|
type InboxWorkItem,
|
|
} from "./inbox";
|
|
|
|
const storage = new Map<string, string>();
|
|
|
|
Object.defineProperty(globalThis, "localStorage", {
|
|
value: {
|
|
getItem: (key: string) => storage.get(key) ?? null,
|
|
setItem: (key: string, value: string) => {
|
|
storage.set(key, value);
|
|
},
|
|
removeItem: (key: string) => {
|
|
storage.delete(key);
|
|
},
|
|
clear: () => {
|
|
storage.clear();
|
|
},
|
|
},
|
|
configurable: true,
|
|
});
|
|
|
|
function makeApproval(status: Approval["status"]): Approval {
|
|
return {
|
|
id: `approval-${status}`,
|
|
companyId: "company-1",
|
|
type: "hire_agent",
|
|
requestedByAgentId: null,
|
|
requestedByUserId: null,
|
|
status,
|
|
payload: {},
|
|
decisionNote: null,
|
|
decidedByUserId: null,
|
|
decidedAt: null,
|
|
createdAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
updatedAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
};
|
|
}
|
|
|
|
function makeApprovalWithTimestamps(
|
|
id: string,
|
|
status: Approval["status"],
|
|
updatedAt: string,
|
|
): Approval {
|
|
return {
|
|
...makeApproval(status),
|
|
id,
|
|
createdAt: new Date(updatedAt),
|
|
updatedAt: new Date(updatedAt),
|
|
};
|
|
}
|
|
|
|
function makeJoinRequest(id: string): JoinRequest {
|
|
return {
|
|
id,
|
|
inviteId: "invite-1",
|
|
companyId: "company-1",
|
|
requestType: "human",
|
|
status: "pending_approval",
|
|
requestEmailSnapshot: null,
|
|
requestIp: "127.0.0.1",
|
|
requestingUserId: null,
|
|
agentName: null,
|
|
adapterType: null,
|
|
capabilities: null,
|
|
agentDefaultsPayload: null,
|
|
claimSecretExpiresAt: null,
|
|
claimSecretConsumedAt: null,
|
|
createdAgentId: null,
|
|
approvedByUserId: null,
|
|
approvedAt: null,
|
|
rejectedByUserId: null,
|
|
rejectedAt: null,
|
|
createdAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
updatedAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
};
|
|
}
|
|
|
|
function makeRun(id: string, status: HeartbeatRun["status"], createdAt: string, agentId = "agent-1"): HeartbeatRun {
|
|
return {
|
|
id,
|
|
companyId: "company-1",
|
|
agentId,
|
|
invocationSource: "assignment",
|
|
triggerDetail: null,
|
|
status,
|
|
error: null,
|
|
wakeupRequestId: null,
|
|
exitCode: null,
|
|
signal: null,
|
|
usageJson: null,
|
|
resultJson: null,
|
|
sessionIdBefore: null,
|
|
sessionIdAfter: null,
|
|
logStore: null,
|
|
logRef: null,
|
|
logBytes: null,
|
|
logSha256: null,
|
|
logCompressed: false,
|
|
errorCode: null,
|
|
externalRunId: null,
|
|
processPid: null,
|
|
processGroupId: null,
|
|
processStartedAt: null,
|
|
retryOfRunId: null,
|
|
processLossRetryCount: 0,
|
|
stdoutExcerpt: null,
|
|
stderrExcerpt: null,
|
|
contextSnapshot: null,
|
|
startedAt: new Date(createdAt),
|
|
finishedAt: null,
|
|
createdAt: new Date(createdAt),
|
|
updatedAt: new Date(createdAt),
|
|
};
|
|
}
|
|
|
|
function makeIssue(id: string, isUnreadForMe: boolean): Issue {
|
|
return {
|
|
id,
|
|
companyId: "company-1",
|
|
projectId: null,
|
|
projectWorkspaceId: null,
|
|
goalId: null,
|
|
parentId: null,
|
|
title: `Issue ${id}`,
|
|
description: null,
|
|
status: "todo",
|
|
priority: "medium",
|
|
assigneeAgentId: null,
|
|
assigneeUserId: null,
|
|
createdByAgentId: null,
|
|
createdByUserId: null,
|
|
issueNumber: 1,
|
|
identifier: `PAP-${id}`,
|
|
requestDepth: 0,
|
|
billingCode: null,
|
|
assigneeAdapterOverrides: null,
|
|
executionWorkspaceId: null,
|
|
executionWorkspacePreference: null,
|
|
executionWorkspaceSettings: null,
|
|
checkoutRunId: null,
|
|
executionRunId: null,
|
|
executionAgentNameKey: null,
|
|
executionLockedAt: null,
|
|
startedAt: null,
|
|
completedAt: null,
|
|
cancelledAt: null,
|
|
hiddenAt: null,
|
|
createdAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
updatedAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
labels: [],
|
|
labelIds: [],
|
|
myLastTouchAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
lastExternalCommentAt: new Date("2026-03-11T01:00:00.000Z"),
|
|
lastActivityAt: new Date("2026-03-11T01:00:00.000Z"),
|
|
isUnreadForMe,
|
|
};
|
|
}
|
|
|
|
function makeProjectWorkspace(overrides: Partial<ProjectWorkspace> = {}): ProjectWorkspace {
|
|
return {
|
|
id: "project-workspace-1",
|
|
companyId: "company-1",
|
|
projectId: "project-1",
|
|
name: "Primary workspace",
|
|
sourceType: "local_path",
|
|
cwd: "/tmp/project",
|
|
repoUrl: null,
|
|
repoRef: null,
|
|
defaultRef: null,
|
|
visibility: "default",
|
|
setupCommand: null,
|
|
cleanupCommand: null,
|
|
remoteProvider: null,
|
|
remoteWorkspaceRef: null,
|
|
sharedWorkspaceKey: null,
|
|
metadata: null,
|
|
runtimeConfig: null,
|
|
isPrimary: true,
|
|
createdAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
updatedAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
function makeExecutionWorkspace(overrides: Partial<ExecutionWorkspace> = {}): ExecutionWorkspace {
|
|
return {
|
|
id: "execution-workspace-1",
|
|
companyId: "company-1",
|
|
projectId: "project-1",
|
|
projectWorkspaceId: "project-workspace-1",
|
|
sourceIssueId: "issue-1",
|
|
mode: "isolated_workspace",
|
|
strategyType: "git_worktree",
|
|
name: "PAP-1 branch",
|
|
status: "active",
|
|
cwd: "/tmp/project/worktree",
|
|
repoUrl: null,
|
|
baseRef: null,
|
|
branchName: "pap-1",
|
|
providerType: "git_worktree",
|
|
providerRef: null,
|
|
derivedFromExecutionWorkspaceId: null,
|
|
lastUsedAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
openedAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
closedAt: null,
|
|
cleanupEligibleAt: null,
|
|
cleanupReason: null,
|
|
config: null,
|
|
metadata: null,
|
|
createdAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
updatedAt: new Date("2026-03-11T00:00:00.000Z"),
|
|
...overrides,
|
|
};
|
|
}
|
|
|
|
const dashboard: DashboardSummary = {
|
|
companyId: "company-1",
|
|
agents: {
|
|
active: 1,
|
|
running: 0,
|
|
paused: 0,
|
|
error: 1,
|
|
},
|
|
tasks: {
|
|
open: 1,
|
|
inProgress: 0,
|
|
blocked: 0,
|
|
done: 0,
|
|
},
|
|
costs: {
|
|
monthSpendCents: 900,
|
|
monthBudgetCents: 1000,
|
|
monthUtilizationPercent: 90,
|
|
},
|
|
pendingApprovals: 1,
|
|
budgets: {
|
|
activeIncidents: 0,
|
|
pendingApprovals: 0,
|
|
pausedAgents: 0,
|
|
pausedProjects: 0,
|
|
},
|
|
};
|
|
|
|
describe("inbox helpers", () => {
|
|
beforeEach(() => {
|
|
storage.clear();
|
|
});
|
|
|
|
it("counts the same inbox sources the badge uses", () => {
|
|
const result = computeInboxBadgeData({
|
|
approvals: [makeApproval("pending"), makeApproval("approved")],
|
|
joinRequests: [makeJoinRequest("join-1")],
|
|
dashboard,
|
|
heartbeatRuns: [
|
|
makeRun("run-old", "failed", "2026-03-11T00:00:00.000Z"),
|
|
makeRun("run-latest", "timed_out", "2026-03-11T01:00:00.000Z"),
|
|
makeRun("run-other-agent", "failed", "2026-03-11T02:00:00.000Z", "agent-2"),
|
|
],
|
|
mineIssues: [makeIssue("1", true)],
|
|
dismissedAlerts: new Set<string>(),
|
|
dismissedAtByKey: new Map<string, number>(),
|
|
});
|
|
|
|
expect(result).toEqual({
|
|
inbox: 6,
|
|
approvals: 1,
|
|
failedRuns: 2,
|
|
joinRequests: 1,
|
|
mineIssues: 1,
|
|
alerts: 1,
|
|
});
|
|
});
|
|
|
|
it("drops dismissed runs and alerts from the computed badge", () => {
|
|
const result = computeInboxBadgeData({
|
|
approvals: [],
|
|
joinRequests: [],
|
|
dashboard,
|
|
heartbeatRuns: [makeRun("run-1", "failed", "2026-03-11T00:00:00.000Z")],
|
|
mineIssues: [],
|
|
dismissedAlerts: new Set<string>(["alert:budget", "alert:agent-errors"]),
|
|
dismissedAtByKey: new Map<string, number>([["run:run-1", new Date("2026-03-11T00:00:00.000Z").getTime()]]),
|
|
});
|
|
|
|
expect(result).toEqual({
|
|
inbox: 0,
|
|
approvals: 0,
|
|
failedRuns: 0,
|
|
joinRequests: 0,
|
|
mineIssues: 0,
|
|
alerts: 0,
|
|
});
|
|
});
|
|
|
|
it("excludes read mine issues from the inbox badge count", () => {
|
|
const result = computeInboxBadgeData({
|
|
approvals: [],
|
|
joinRequests: [],
|
|
dashboard,
|
|
heartbeatRuns: [],
|
|
mineIssues: [makeIssue("1", false), makeIssue("2", false), makeIssue("3", true)],
|
|
dismissedAlerts: new Set<string>(),
|
|
dismissedAtByKey: new Map(),
|
|
});
|
|
|
|
expect(result.mineIssues).toBe(1);
|
|
expect(result.inbox).toBe(3);
|
|
});
|
|
|
|
it("resurfaces non-issue items when they change after dismissal", () => {
|
|
const dismissedAtByKey = buildInboxDismissedAtByKey([
|
|
{
|
|
id: "dismissal-1",
|
|
companyId: "company-1",
|
|
userId: "user-1",
|
|
itemKey: "approval:approval-1",
|
|
dismissedAt: new Date("2026-03-11T01:00:00.000Z"),
|
|
createdAt: new Date("2026-03-11T01:00:00.000Z"),
|
|
updatedAt: new Date("2026-03-11T01:00:00.000Z"),
|
|
},
|
|
]);
|
|
|
|
expect(
|
|
isInboxEntityDismissed(
|
|
dismissedAtByKey,
|
|
"approval:approval-1",
|
|
new Date("2026-03-11T00:30:00.000Z"),
|
|
),
|
|
).toBe(true);
|
|
expect(
|
|
isInboxEntityDismissed(
|
|
dismissedAtByKey,
|
|
"approval:approval-1",
|
|
new Date("2026-03-11T01:30:00.000Z"),
|
|
),
|
|
).toBe(false);
|
|
});
|
|
|
|
it("keeps read issues in the touched list but excludes them from unread counts", () => {
|
|
const issues = [makeIssue("1", true), makeIssue("2", false)];
|
|
|
|
expect(getUnreadTouchedIssues(issues).map((issue) => issue.id)).toEqual(["1"]);
|
|
expect(issues).toHaveLength(2);
|
|
});
|
|
|
|
it("shows recent approvals in updated order and unread approvals as actionable only", () => {
|
|
const approvals = [
|
|
makeApprovalWithTimestamps("approval-approved", "approved", "2026-03-11T02:00:00.000Z"),
|
|
makeApprovalWithTimestamps("approval-pending", "pending", "2026-03-11T01:00:00.000Z"),
|
|
makeApprovalWithTimestamps(
|
|
"approval-revision",
|
|
"revision_requested",
|
|
"2026-03-11T03:00:00.000Z",
|
|
),
|
|
];
|
|
|
|
expect(getApprovalsForTab(approvals, "mine", "all").map((approval) => approval.id)).toEqual([
|
|
"approval-revision",
|
|
"approval-approved",
|
|
"approval-pending",
|
|
]);
|
|
expect(getApprovalsForTab(approvals, "recent", "all").map((approval) => approval.id)).toEqual([
|
|
"approval-revision",
|
|
"approval-approved",
|
|
"approval-pending",
|
|
]);
|
|
expect(getApprovalsForTab(approvals, "unread", "all").map((approval) => approval.id)).toEqual([
|
|
"approval-revision",
|
|
"approval-pending",
|
|
]);
|
|
expect(getApprovalsForTab(approvals, "all", "resolved").map((approval) => approval.id)).toEqual([
|
|
"approval-approved",
|
|
]);
|
|
});
|
|
|
|
it("mixes approvals into the inbox feed by most recent activity", () => {
|
|
const newerIssue = makeIssue("1", true);
|
|
newerIssue.lastActivityAt = new Date("2026-03-11T04:00:00.000Z");
|
|
|
|
const olderIssue = makeIssue("2", false);
|
|
olderIssue.lastActivityAt = new Date("2026-03-11T02:00:00.000Z");
|
|
|
|
const approval = makeApprovalWithTimestamps(
|
|
"approval-between",
|
|
"pending",
|
|
"2026-03-11T03:00:00.000Z",
|
|
);
|
|
|
|
expect(
|
|
getInboxWorkItems({
|
|
issues: [olderIssue, newerIssue],
|
|
approvals: [approval],
|
|
}).map((item) => {
|
|
if (item.kind === "issue") return `issue:${item.issue.id}`;
|
|
if (item.kind === "approval") return `approval:${item.approval.id}`;
|
|
if (item.kind === "join_request") return `join:${item.joinRequest.id}`;
|
|
return `run:${item.run.id}`;
|
|
}),
|
|
).toEqual([
|
|
"issue:1",
|
|
"approval:approval-between",
|
|
"issue:2",
|
|
]);
|
|
});
|
|
|
|
it("prefers canonical lastActivityAt over comment-only timestamps", () => {
|
|
const activityIssue = makeIssue("1", true);
|
|
activityIssue.lastExternalCommentAt = new Date("2026-03-11T01:00:00.000Z");
|
|
activityIssue.lastActivityAt = new Date("2026-03-11T05:00:00.000Z");
|
|
|
|
const commentIssue = makeIssue("2", true);
|
|
commentIssue.lastExternalCommentAt = new Date("2026-03-11T04:00:00.000Z");
|
|
commentIssue.lastActivityAt = new Date("2026-03-11T04:00:00.000Z");
|
|
|
|
expect(getRecentTouchedIssues([commentIssue, activityIssue]).map((issue) => issue.id)).toEqual(["1", "2"]);
|
|
});
|
|
|
|
it("mixes join requests into the inbox feed by most recent activity", () => {
|
|
const issue = makeIssue("1", true);
|
|
issue.lastActivityAt = new Date("2026-03-11T04:00:00.000Z");
|
|
|
|
const joinRequest = makeJoinRequest("join-1");
|
|
joinRequest.createdAt = new Date("2026-03-11T03:00:00.000Z");
|
|
|
|
const approval = makeApprovalWithTimestamps(
|
|
"approval-oldest",
|
|
"pending",
|
|
"2026-03-11T02:00:00.000Z",
|
|
);
|
|
|
|
expect(
|
|
getInboxWorkItems({
|
|
issues: [issue],
|
|
approvals: [approval],
|
|
joinRequests: [joinRequest],
|
|
}).map((item) => {
|
|
if (item.kind === "issue") return `issue:${item.issue.id}`;
|
|
if (item.kind === "approval") return `approval:${item.approval.id}`;
|
|
if (item.kind === "join_request") return `join:${item.joinRequest.id}`;
|
|
return `run:${item.run.id}`;
|
|
}),
|
|
).toEqual([
|
|
"issue:1",
|
|
"join:join-1",
|
|
"approval:approval-oldest",
|
|
]);
|
|
});
|
|
|
|
it("skips hidden groups when building keyboard navigation entries", () => {
|
|
const visibleIssue = makeIssue("visible", true);
|
|
const hiddenIssue = makeIssue("hidden", true);
|
|
const approval = makeApprovalWithTimestamps("approval-1", "pending", "2026-03-11T03:00:00.000Z");
|
|
|
|
const entries = buildInboxKeyboardNavEntries(
|
|
[
|
|
{
|
|
key: "visible-group",
|
|
displayItems: [{ kind: "issue", timestamp: 3, issue: visibleIssue }],
|
|
childrenByIssueId: new Map(),
|
|
},
|
|
{
|
|
key: "hidden-group",
|
|
displayItems: [
|
|
{ kind: "issue", timestamp: 2, issue: hiddenIssue },
|
|
{ kind: "approval", timestamp: 1, approval },
|
|
],
|
|
childrenByIssueId: new Map(),
|
|
},
|
|
],
|
|
new Set(["hidden-group"]),
|
|
new Set(),
|
|
);
|
|
|
|
expect(entries).toEqual([
|
|
{
|
|
type: "top",
|
|
itemKey: `visible-group:${getInboxWorkItemKey({ kind: "issue", timestamp: 3, issue: visibleIssue })}`,
|
|
item: { kind: "issue", timestamp: 3, issue: visibleIssue },
|
|
},
|
|
]);
|
|
});
|
|
|
|
it("includes child issues only when their parent row is expanded", () => {
|
|
const parentIssue = makeIssue("parent", true);
|
|
const childIssue = makeIssue("child", true);
|
|
childIssue.parentId = parentIssue.id;
|
|
|
|
const groupedSections = [
|
|
{
|
|
key: "workspace:default",
|
|
displayItems: [{ kind: "issue", timestamp: 2, issue: parentIssue } satisfies InboxWorkItem],
|
|
childrenByIssueId: new Map([[parentIssue.id, [childIssue]]]),
|
|
},
|
|
];
|
|
|
|
expect(
|
|
buildInboxKeyboardNavEntries(groupedSections, new Set(), new Set()).map((entry) => entry.type === "top"
|
|
? entry.itemKey
|
|
: entry.issueId),
|
|
).toEqual([
|
|
`workspace:default:${getInboxWorkItemKey({ kind: "issue", timestamp: 2, issue: parentIssue })}`,
|
|
childIssue.id,
|
|
]);
|
|
|
|
expect(
|
|
buildInboxKeyboardNavEntries(groupedSections, new Set(), new Set([parentIssue.id])).map((entry) => entry.type === "top"
|
|
? entry.itemKey
|
|
: entry.issueId),
|
|
).toEqual([
|
|
`workspace:default:${getInboxWorkItemKey({ kind: "issue", timestamp: 2, issue: parentIssue })}`,
|
|
]);
|
|
});
|
|
|
|
it("sorts self-touched issues without external comments by updatedAt", () => {
|
|
const recentSelfTouched = makeIssue("recent", false);
|
|
recentSelfTouched.lastExternalCommentAt = null as unknown as Date;
|
|
recentSelfTouched.updatedAt = new Date("2026-03-11T05:00:00.000Z");
|
|
recentSelfTouched.myLastTouchAt = new Date("2026-03-11T05:00:00.000Z");
|
|
|
|
const olderCommented = makeIssue("older", false);
|
|
olderCommented.lastExternalCommentAt = new Date("2026-03-11T03:00:00.000Z");
|
|
|
|
const items = getInboxWorkItems({
|
|
issues: [olderCommented, recentSelfTouched],
|
|
approvals: [],
|
|
});
|
|
|
|
expect(items.map((item) => (item.kind === "issue" ? item.issue.id : ""))).toEqual([
|
|
"recent",
|
|
"older",
|
|
]);
|
|
});
|
|
|
|
it("can include sections on recent without forcing them to be unread", () => {
|
|
expect(
|
|
shouldShowInboxSection({
|
|
tab: "mine",
|
|
hasItems: true,
|
|
showOnMine: true,
|
|
showOnRecent: false,
|
|
showOnUnread: false,
|
|
showOnAll: false,
|
|
}),
|
|
).toBe(true);
|
|
expect(
|
|
shouldShowInboxSection({
|
|
tab: "recent",
|
|
hasItems: true,
|
|
showOnMine: false,
|
|
showOnRecent: true,
|
|
showOnUnread: false,
|
|
showOnAll: false,
|
|
}),
|
|
).toBe(true);
|
|
expect(
|
|
shouldShowInboxSection({
|
|
tab: "unread",
|
|
hasItems: true,
|
|
showOnMine: true,
|
|
showOnRecent: true,
|
|
showOnUnread: false,
|
|
showOnAll: false,
|
|
}),
|
|
).toBe(false);
|
|
});
|
|
|
|
it("limits recent touched issues before unread badge counting", () => {
|
|
const issues = Array.from({ length: RECENT_ISSUES_LIMIT + 5 }, (_, index) => {
|
|
const issue = makeIssue(String(index + 1), index < 3);
|
|
issue.lastActivityAt = new Date(Date.UTC(2026, 2, 31, 0, 0, 0, 0) - index * 60_000);
|
|
return issue;
|
|
});
|
|
|
|
const recentIssues = getRecentTouchedIssues(issues);
|
|
|
|
expect(recentIssues).toHaveLength(RECENT_ISSUES_LIMIT);
|
|
expect(getUnreadTouchedIssues(recentIssues).map((issue) => issue.id)).toEqual(["1", "2", "3"]);
|
|
});
|
|
|
|
it("matches workspace names when inbox issue search includes workspace labels", () => {
|
|
const issue = makeIssue("workspace", false);
|
|
issue.projectId = "project-1";
|
|
issue.projectWorkspaceId = "project-workspace-1";
|
|
issue.executionWorkspaceId = "execution-workspace-1";
|
|
|
|
expect(matchesInboxIssueSearch(
|
|
issue,
|
|
"feature",
|
|
{
|
|
isolatedWorkspacesEnabled: true,
|
|
executionWorkspaceById: new Map([
|
|
["execution-workspace-1", { name: "Feature Branch", mode: "isolated_workspace" as const, projectWorkspaceId: "project-workspace-1" }],
|
|
]),
|
|
projectWorkspaceById: new Map([
|
|
["project-workspace-1", { name: "Primary workspace" }],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([["project-1", "project-workspace-2"]]),
|
|
},
|
|
)).toBe(true);
|
|
});
|
|
|
|
it("resolves the default workspace into an explicit grouping label", () => {
|
|
const issue = makeIssue("default", false);
|
|
issue.projectId = "project-1";
|
|
issue.projectWorkspaceId = "project-workspace-1";
|
|
|
|
expect(resolveIssueWorkspaceGroup(issue, {
|
|
projectWorkspaceById: new Map([
|
|
["project-workspace-1", { name: "Primary workspace" }],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([["project-1", "project-workspace-1"]]),
|
|
})).toEqual({
|
|
key: "workspace:project:project-workspace-1",
|
|
label: "Primary workspace (default)",
|
|
});
|
|
});
|
|
|
|
it("returns archived search matches that are not already visible in the inbox", () => {
|
|
const visibleIssue = makeIssue("visible", false);
|
|
visibleIssue.title = "Alpha visible task";
|
|
|
|
const archivedMatch = makeIssue("archived-match", false);
|
|
archivedMatch.title = "Alpha archived task";
|
|
|
|
const archivedMiss = makeIssue("archived-miss", false);
|
|
archivedMiss.title = "Different task";
|
|
|
|
expect(
|
|
getArchivedInboxSearchIssues({
|
|
visibleIssues: [visibleIssue],
|
|
searchableIssues: [visibleIssue, archivedMatch, archivedMiss],
|
|
query: "alpha",
|
|
}).map((issue) => issue.id),
|
|
).toEqual(["archived-match"]);
|
|
});
|
|
|
|
it("sorts archived search matches by most recent activity", () => {
|
|
const older = makeIssue("older", false);
|
|
older.title = "Alpha older";
|
|
older.lastActivityAt = new Date("2026-03-11T02:00:00.000Z");
|
|
|
|
const newer = makeIssue("newer", false);
|
|
newer.title = "Alpha newer";
|
|
newer.lastActivityAt = new Date("2026-03-11T03:00:00.000Z");
|
|
|
|
expect(
|
|
getArchivedInboxSearchIssues({
|
|
visibleIssues: [],
|
|
searchableIssues: [older, newer],
|
|
query: "alpha",
|
|
}).map((issue) => issue.id),
|
|
).toEqual(["newer", "older"]);
|
|
});
|
|
|
|
it("adds remote issue results that are not already present in inbox search results", () => {
|
|
const remoteMatch = makeIssue("remote-match", false);
|
|
remoteMatch.status = "in_progress";
|
|
|
|
expect(
|
|
getInboxSearchSupplementIssues({
|
|
query: "pull/3303",
|
|
filteredWorkItems: [],
|
|
archivedSearchIssues: [],
|
|
remoteIssues: [remoteMatch],
|
|
issueFilters: {
|
|
statuses: ["in_progress"],
|
|
priorities: [],
|
|
assignees: [],
|
|
creators: [],
|
|
labels: [],
|
|
projects: [],
|
|
workspaces: [],
|
|
hideRoutineExecutions: true,
|
|
},
|
|
}).map((issue) => issue.id),
|
|
).toEqual(["remote-match"]);
|
|
|
|
expect(
|
|
getInboxSearchSupplementIssues({
|
|
query: "pull/3303",
|
|
filteredWorkItems: [{ kind: "issue", timestamp: 1, issue: makeIssue("remote-match", false) }],
|
|
archivedSearchIssues: [],
|
|
remoteIssues: [remoteMatch],
|
|
issueFilters: {
|
|
statuses: [],
|
|
priorities: [],
|
|
assignees: [],
|
|
creators: [],
|
|
labels: [],
|
|
projects: [],
|
|
workspaces: [],
|
|
hideRoutineExecutions: true,
|
|
},
|
|
}),
|
|
).toEqual([]);
|
|
|
|
expect(
|
|
getInboxSearchSupplementIssues({
|
|
query: "pull/3303",
|
|
filteredWorkItems: [],
|
|
archivedSearchIssues: [makeIssue("remote-match", false)],
|
|
remoteIssues: [remoteMatch],
|
|
issueFilters: {
|
|
statuses: [],
|
|
priorities: [],
|
|
assignees: [],
|
|
creators: [],
|
|
labels: [],
|
|
projects: [],
|
|
workspaces: [],
|
|
hideRoutineExecutions: true,
|
|
},
|
|
}),
|
|
).toEqual([]);
|
|
});
|
|
|
|
it("keeps inbox search matches ahead of archived and other result sections", () => {
|
|
const inboxIssue = makeIssue("inbox", false);
|
|
inboxIssue.lastActivityAt = new Date("2026-03-11T04:00:00.000Z");
|
|
|
|
const archivedIssue = makeIssue("archived", false);
|
|
archivedIssue.lastActivityAt = new Date("2026-03-11T03:00:00.000Z");
|
|
|
|
const otherIssue = makeIssue("other", false);
|
|
otherIssue.lastActivityAt = new Date("2026-03-11T05:00:00.000Z");
|
|
|
|
const sections = [
|
|
...buildGroupedInboxSections(
|
|
getInboxWorkItems({ issues: [inboxIssue], approvals: [] }),
|
|
"none",
|
|
{},
|
|
),
|
|
...buildGroupedInboxSections(
|
|
getInboxWorkItems({ issues: [archivedIssue], approvals: [] }),
|
|
"none",
|
|
{},
|
|
{ keyPrefix: "archived-search:", searchSection: "archived" },
|
|
),
|
|
...buildGroupedInboxSections(
|
|
getInboxWorkItems({ issues: [otherIssue], approvals: [] }),
|
|
"none",
|
|
{},
|
|
{ keyPrefix: "other-search:", searchSection: "other" },
|
|
),
|
|
];
|
|
|
|
expect(sections.map((section) => section.searchSection)).toEqual(["none", "archived", "other"]);
|
|
expect(
|
|
sections.map((section) => {
|
|
const [item] = section.displayItems;
|
|
return item?.kind === "issue" ? item.issue.id : null;
|
|
}),
|
|
).toEqual(["inbox", "archived", "other"]);
|
|
});
|
|
|
|
it("defaults the remembered inbox tab to mine and persists all", () => {
|
|
localStorage.clear();
|
|
expect(loadLastInboxTab()).toBe("mine");
|
|
|
|
saveLastInboxTab("all");
|
|
expect(loadLastInboxTab()).toBe("all");
|
|
});
|
|
|
|
it("persists inbox filters per company", () => {
|
|
saveInboxFilterPreferences("company-1", {
|
|
allCategoryFilter: "approvals",
|
|
allApprovalFilter: "resolved",
|
|
issueFilters: {
|
|
statuses: ["todo"],
|
|
priorities: ["high"],
|
|
assignees: ["agent-1"],
|
|
creators: ["user:user-1"],
|
|
labels: ["label-1"],
|
|
projects: ["project-1"],
|
|
workspaces: ["workspace-1"],
|
|
hideRoutineExecutions: false,
|
|
},
|
|
});
|
|
saveInboxFilterPreferences("company-2", {
|
|
allCategoryFilter: "failed_runs",
|
|
allApprovalFilter: "actionable",
|
|
issueFilters: {
|
|
statuses: ["done"],
|
|
priorities: [],
|
|
assignees: [],
|
|
creators: [],
|
|
labels: [],
|
|
projects: [],
|
|
workspaces: [],
|
|
hideRoutineExecutions: true,
|
|
},
|
|
});
|
|
|
|
expect(loadInboxFilterPreferences("company-1")).toEqual({
|
|
allCategoryFilter: "approvals",
|
|
allApprovalFilter: "resolved",
|
|
issueFilters: {
|
|
statuses: ["todo"],
|
|
priorities: ["high"],
|
|
assignees: ["agent-1"],
|
|
creators: ["user:user-1"],
|
|
labels: ["label-1"],
|
|
projects: ["project-1"],
|
|
workspaces: ["workspace-1"],
|
|
hideRoutineExecutions: false,
|
|
},
|
|
});
|
|
expect(loadInboxFilterPreferences("company-2")).toEqual({
|
|
allCategoryFilter: "failed_runs",
|
|
allApprovalFilter: "actionable",
|
|
issueFilters: {
|
|
statuses: ["done"],
|
|
priorities: [],
|
|
assignees: [],
|
|
creators: [],
|
|
labels: [],
|
|
projects: [],
|
|
workspaces: [],
|
|
hideRoutineExecutions: true,
|
|
},
|
|
});
|
|
});
|
|
|
|
it("normalizes invalid inbox filter storage back to safe defaults", () => {
|
|
localStorage.setItem("paperclip:inbox:filters:company-1", JSON.stringify({
|
|
allCategoryFilter: "bogus",
|
|
allApprovalFilter: "bogus",
|
|
issueFilters: {
|
|
statuses: ["todo", 123],
|
|
priorities: "high",
|
|
assignees: ["agent-1"],
|
|
creators: ["user:user-1", 42],
|
|
labels: null,
|
|
projects: ["project-1"],
|
|
workspaces: ["workspace-1", false],
|
|
hideRoutineExecutions: "yes",
|
|
},
|
|
}));
|
|
|
|
expect(loadInboxFilterPreferences("company-1")).toEqual({
|
|
allCategoryFilter: "everything",
|
|
allApprovalFilter: "all",
|
|
issueFilters: {
|
|
statuses: ["todo"],
|
|
priorities: [],
|
|
assignees: ["agent-1"],
|
|
creators: ["user:user-1"],
|
|
labels: [],
|
|
projects: ["project-1"],
|
|
workspaces: ["workspace-1"],
|
|
hideRoutineExecutions: false,
|
|
},
|
|
});
|
|
});
|
|
|
|
it("keeps nesting enabled on desktop when the saved preference is on", () => {
|
|
expect(resolveInboxNestingEnabled(true, false)).toBe(true);
|
|
});
|
|
|
|
it("forces nesting off on mobile even when the saved preference is on", () => {
|
|
expect(resolveInboxNestingEnabled(true, true)).toBe(false);
|
|
});
|
|
|
|
it("keeps nesting off when the saved preference is off", () => {
|
|
expect(resolveInboxNestingEnabled(false, false)).toBe(false);
|
|
expect(resolveInboxNestingEnabled(false, true)).toBe(false);
|
|
});
|
|
|
|
it("defaults issue columns to the current inbox layout", () => {
|
|
expect(loadInboxIssueColumns()).toEqual(DEFAULT_INBOX_ISSUE_COLUMNS);
|
|
});
|
|
|
|
it("normalizes saved issue columns to valid values in canonical order", () => {
|
|
saveInboxIssueColumns(["labels", "updated", "status", "workspace", "labels", "assignee"]);
|
|
|
|
expect(loadInboxIssueColumns()).toEqual(["status", "assignee", "workspace", "labels", "updated"]);
|
|
expect(normalizeInboxIssueColumns(["project", "workspace", "wat", "id"])).toEqual(["id", "project", "workspace"]);
|
|
});
|
|
|
|
it("hides the workspace column option unless isolated workspaces are enabled", () => {
|
|
expect(getAvailableInboxIssueColumns(false)).toEqual(["status", "id", "assignee", "project", "parent", "labels", "updated"]);
|
|
expect(getAvailableInboxIssueColumns(true)).toEqual([
|
|
"status",
|
|
"id",
|
|
"assignee",
|
|
"project",
|
|
"workspace",
|
|
"parent",
|
|
"labels",
|
|
"updated",
|
|
]);
|
|
});
|
|
|
|
it("allows hiding every optional issue column down to the title-only view", () => {
|
|
saveInboxIssueColumns([]);
|
|
expect(loadInboxIssueColumns()).toEqual([]);
|
|
});
|
|
|
|
it("shows explicit workspace names but leaves the default workspace blank", () => {
|
|
const issue = makeIssue("1", true);
|
|
issue.projectId = "project-1";
|
|
issue.projectWorkspaceId = "project-workspace-1";
|
|
issue.executionWorkspaceId = "execution-workspace-1";
|
|
|
|
const executionWorkspace = makeExecutionWorkspace();
|
|
const defaultWorkspace = makeProjectWorkspace();
|
|
const secondaryWorkspace = makeProjectWorkspace({
|
|
id: "project-workspace-2",
|
|
name: "Secondary workspace",
|
|
isPrimary: false,
|
|
});
|
|
|
|
expect(
|
|
resolveIssueWorkspaceName(issue, {
|
|
executionWorkspaceById: new Map([[executionWorkspace.id, executionWorkspace]]),
|
|
projectWorkspaceById: new Map([
|
|
[defaultWorkspace.id, defaultWorkspace],
|
|
[secondaryWorkspace.id, secondaryWorkspace],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([[issue.projectId!, defaultWorkspace.id]]),
|
|
}),
|
|
).toBe("PAP-1 branch");
|
|
|
|
issue.executionWorkspaceId = null;
|
|
expect(
|
|
resolveIssueWorkspaceName(issue, {
|
|
projectWorkspaceById: new Map([
|
|
[defaultWorkspace.id, defaultWorkspace],
|
|
[secondaryWorkspace.id, secondaryWorkspace],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([[issue.projectId!, defaultWorkspace.id]]),
|
|
}),
|
|
).toBeNull();
|
|
|
|
issue.projectWorkspaceId = secondaryWorkspace.id;
|
|
expect(
|
|
resolveIssueWorkspaceName(issue, {
|
|
projectWorkspaceById: new Map([
|
|
[defaultWorkspace.id, defaultWorkspace],
|
|
[secondaryWorkspace.id, secondaryWorkspace],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([[issue.projectId!, defaultWorkspace.id]]),
|
|
}),
|
|
).toBe("Secondary workspace");
|
|
|
|
issue.projectWorkspaceId = null;
|
|
expect(
|
|
resolveIssueWorkspaceName(issue, {
|
|
projectWorkspaceById: new Map([
|
|
[defaultWorkspace.id, defaultWorkspace],
|
|
[secondaryWorkspace.id, secondaryWorkspace],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([[issue.projectId!, defaultWorkspace.id]]),
|
|
}),
|
|
).toBeNull();
|
|
|
|
issue.executionWorkspaceId = "execution-workspace-shared-default";
|
|
issue.projectWorkspaceId = defaultWorkspace.id;
|
|
expect(
|
|
resolveIssueWorkspaceName(issue, {
|
|
executionWorkspaceById: new Map([[
|
|
issue.executionWorkspaceId,
|
|
makeExecutionWorkspace({
|
|
id: issue.executionWorkspaceId,
|
|
mode: "shared_workspace",
|
|
strategyType: "project_primary",
|
|
projectWorkspaceId: defaultWorkspace.id,
|
|
name: "PAP-1067",
|
|
}),
|
|
]]),
|
|
projectWorkspaceById: new Map([
|
|
[defaultWorkspace.id, defaultWorkspace],
|
|
[secondaryWorkspace.id, secondaryWorkspace],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([[issue.projectId!, defaultWorkspace.id]]),
|
|
}),
|
|
).toBeNull();
|
|
});
|
|
|
|
it("maps legacy new-tab storage to mine", () => {
|
|
localStorage.setItem("paperclip:inbox:last-tab", "new");
|
|
expect(loadLastInboxTab()).toBe("mine");
|
|
});
|
|
|
|
it("enables swipe archive only on the mine tab", () => {
|
|
expect(isMineInboxTab("mine")).toBe(true);
|
|
expect(isMineInboxTab("recent")).toBe(false);
|
|
expect(isMineInboxTab("unread")).toBe(false);
|
|
expect(isMineInboxTab("all")).toBe(false);
|
|
});
|
|
|
|
it("anchors Mine selection to the first available inbox row", () => {
|
|
expect(resolveInboxSelectionIndex(-1, 3)).toBe(-1);
|
|
expect(resolveInboxSelectionIndex(5, 3)).toBe(2);
|
|
expect(resolveInboxSelectionIndex(1, 0)).toBe(-1);
|
|
});
|
|
|
|
it("selects the first row only after keyboard navigation starts", () => {
|
|
expect(getInboxKeyboardSelectionIndex(-1, 3, "next")).toBe(0);
|
|
expect(getInboxKeyboardSelectionIndex(-1, 3, "previous")).toBe(0);
|
|
expect(getInboxKeyboardSelectionIndex(0, 3, "next")).toBe(1);
|
|
expect(getInboxKeyboardSelectionIndex(0, 3, "previous")).toBe(0);
|
|
});
|
|
|
|
it("hides routine execution issues when the hide toggle is enabled", () => {
|
|
const manualIssue = { ...makeIssue("manual", true), originKind: "manual" as const };
|
|
const routineIssue = { ...makeIssue("routine", true), originKind: "routine_execution" as const };
|
|
|
|
expect(filterInboxIssues([manualIssue, routineIssue], false)).toEqual([manualIssue, routineIssue]);
|
|
expect(filterInboxIssues([manualIssue, routineIssue], true)).toEqual([manualIssue]);
|
|
});
|
|
|
|
it("groups mixed inbox items by type while preserving item order within each group", () => {
|
|
const items: InboxWorkItem[] = [
|
|
{ kind: "approval", timestamp: 4, approval: makeApproval("pending") },
|
|
{ kind: "issue", timestamp: 3, issue: makeIssue("1", true) },
|
|
{ kind: "issue", timestamp: 2, issue: makeIssue("2", false) },
|
|
{ kind: "failed_run", timestamp: 1, run: makeRun("run-1", "failed", "2026-03-11T00:00:00.000Z") },
|
|
{ kind: "join_request", timestamp: 0, joinRequest: makeJoinRequest("join-1") },
|
|
];
|
|
|
|
expect(groupInboxWorkItems(items, "none")).toEqual([{ key: "__all", label: null, items }]);
|
|
expect(groupInboxWorkItems(items, "type")).toEqual([
|
|
{ key: "issue", label: "Issues", items: [items[1], items[2]] },
|
|
{ key: "approval", label: "Approvals", items: [items[0]] },
|
|
{ key: "failed_run", label: "Failed runs", items: [items[3]] },
|
|
{ key: "join_request", label: "Join requests", items: [items[4]] },
|
|
]);
|
|
});
|
|
|
|
it("groups workspace sections by latest issue activity while preserving non-issue sections", () => {
|
|
const defaultIssue = makeIssue("default", true);
|
|
defaultIssue.projectId = "project-1";
|
|
defaultIssue.projectWorkspaceId = "project-workspace-1";
|
|
|
|
const sharedDefaultIssue = makeIssue("shared-default", true);
|
|
sharedDefaultIssue.projectId = "project-1";
|
|
sharedDefaultIssue.projectWorkspaceId = "project-workspace-1";
|
|
sharedDefaultIssue.executionWorkspaceId = "execution-workspace-shared-default";
|
|
|
|
const featureIssue = makeIssue("feature", false);
|
|
featureIssue.projectId = "project-1";
|
|
featureIssue.projectWorkspaceId = "project-workspace-2";
|
|
|
|
const execIssue = makeIssue("exec", false);
|
|
execIssue.projectId = "project-1";
|
|
execIssue.projectWorkspaceId = "project-workspace-1";
|
|
execIssue.executionWorkspaceId = "execution-workspace-1";
|
|
|
|
const items: InboxWorkItem[] = [
|
|
{ kind: "issue", timestamp: 5, issue: defaultIssue },
|
|
{ kind: "approval", timestamp: 2, approval: makeApproval("pending") },
|
|
{ kind: "issue", timestamp: 4, issue: sharedDefaultIssue },
|
|
{ kind: "issue", timestamp: 7, issue: featureIssue },
|
|
{ kind: "issue", timestamp: 9, issue: execIssue },
|
|
];
|
|
|
|
expect(groupInboxWorkItems(items, "workspace", {
|
|
executionWorkspaceById: new Map([
|
|
["execution-workspace-1", { name: "Feature Branch", mode: "isolated_workspace", projectWorkspaceId: "project-workspace-1" }],
|
|
["execution-workspace-shared-default", { name: "Shared default workspace", mode: "shared_workspace", projectWorkspaceId: "project-workspace-1" }],
|
|
]),
|
|
projectWorkspaceById: new Map([
|
|
["project-workspace-1", { name: "Primary workspace" }],
|
|
["project-workspace-2", { name: "Secondary workspace" }],
|
|
]),
|
|
defaultProjectWorkspaceIdByProjectId: new Map([["project-1", "project-workspace-1"]]),
|
|
})).toEqual([
|
|
{ key: "workspace:execution:execution-workspace-1", label: "Feature Branch", items: [items[4]] },
|
|
{ key: "workspace:project:project-workspace-2", label: "Secondary workspace", items: [items[3]] },
|
|
{
|
|
key: "workspace:project:project-workspace-1",
|
|
label: "Primary workspace (default)",
|
|
items: [items[0], items[2]],
|
|
},
|
|
{ key: "kind:approval", label: "Approvals", items: [items[1]] },
|
|
]);
|
|
});
|
|
|
|
it("persists workspace grouping preferences", () => {
|
|
saveInboxWorkItemGroupBy("workspace");
|
|
expect(loadInboxWorkItemGroupBy()).toBe("workspace");
|
|
});
|
|
|
|
it("persists collapsed inbox groups per company", () => {
|
|
saveCollapsedInboxGroupKeys("company-1", new Set(["workspace:alpha", "workspace:beta"]));
|
|
saveCollapsedInboxGroupKeys("company-2", new Set(["type:approval"]));
|
|
|
|
expect(loadCollapsedInboxGroupKeys("company-1")).toEqual(new Set(["workspace:alpha", "workspace:beta"]));
|
|
expect(loadCollapsedInboxGroupKeys("company-2")).toEqual(new Set(["type:approval"]));
|
|
});
|
|
|
|
it("returns empty collapsed inbox groups for missing or invalid storage", () => {
|
|
expect(loadCollapsedInboxGroupKeys("company-1")).toEqual(new Set());
|
|
localStorage.setItem("paperclip:inbox:collapsed-groups:company-1", JSON.stringify({ nope: true }));
|
|
expect(loadCollapsedInboxGroupKeys("company-1")).toEqual(new Set());
|
|
});
|
|
|
|
it("does not reset workspace grouping before experimental settings have loaded", () => {
|
|
expect(shouldResetInboxWorkspaceGrouping("workspace", false, false)).toBe(false);
|
|
});
|
|
|
|
it("resets workspace grouping only when settings are loaded and workspace grouping is unavailable", () => {
|
|
expect(shouldResetInboxWorkspaceGrouping("workspace", false, true)).toBe(true);
|
|
expect(shouldResetInboxWorkspaceGrouping("workspace", true, true)).toBe(false);
|
|
expect(shouldResetInboxWorkspaceGrouping("none", false, true)).toBe(false);
|
|
});
|
|
});
|