Implement assistant-ui issue chat thread

Co-Authored-By: Paperclip <noreply@paperclip.ing>
This commit is contained in:
dotta 2026-04-06 07:53:37 -05:00
parent 9cfa37fce3
commit 73abe4c76e
6 changed files with 1647 additions and 24 deletions

View file

@ -0,0 +1,179 @@
import { describe, expect, it } from "vitest";
import type { Agent } from "@paperclipai/shared";
import {
buildAssistantPartsFromTranscript,
buildIssueChatMessages,
type IssueChatComment,
type IssueChatLinkedRun,
} from "./issue-chat-messages";
import type { IssueTimelineEvent } from "./issue-timeline-events";
import type { LiveRunForIssue } from "../api/heartbeats";
function createAgent(id: string, name: string): Agent {
return {
id,
companyId: "company-1",
name,
role: "engineer",
title: null,
icon: "code",
status: "active",
reportsTo: null,
capabilities: null,
adapterType: "codex_local",
adapterConfig: {},
runtimeConfig: {},
budgetMonthlyCents: 0,
spentMonthlyCents: 0,
lastHeartbeatAt: null,
metadata: {},
createdAt: new Date(),
updatedAt: new Date(),
pauseReason: null,
pausedAt: null,
urlKey: "codexcoder",
permissions: { canCreateAgents: false },
} as Agent;
}
function createComment(overrides: Partial<IssueChatComment> = {}): IssueChatComment {
return {
id: "comment-1",
companyId: "company-1",
issueId: "issue-1",
authorAgentId: null,
authorUserId: "user-1",
body: "Hello",
createdAt: new Date("2026-04-06T12:00:00.000Z"),
updatedAt: new Date("2026-04-06T12:00:00.000Z"),
...overrides,
};
}
describe("buildAssistantPartsFromTranscript", () => {
it("maps assistant text, reasoning, tool calls, and tool results", () => {
const result = buildAssistantPartsFromTranscript([
{ kind: "assistant", ts: "2026-04-06T12:00:00.000Z", text: "Working on it. " },
{ kind: "assistant", ts: "2026-04-06T12:00:01.000Z", text: "Done." },
{ kind: "thinking", ts: "2026-04-06T12:00:02.000Z", text: "Need to inspect files." },
{
kind: "tool_call",
ts: "2026-04-06T12:00:03.000Z",
name: "read_file",
toolUseId: "tool-1",
input: { path: "ui/src/pages/IssueDetail.tsx" },
},
{
kind: "tool_result",
ts: "2026-04-06T12:00:04.000Z",
toolUseId: "tool-1",
content: "file contents",
isError: false,
},
{ kind: "stderr", ts: "2026-04-06T12:00:05.000Z", text: "warn: noisy setup output" },
]);
expect(result.parts).toHaveLength(3);
expect(result.parts[0]).toMatchObject({ type: "text", text: "Working on it. Done." });
expect(result.parts[1]).toMatchObject({ type: "reasoning", text: "Need to inspect files." });
expect(result.parts[2]).toMatchObject({
type: "tool-call",
toolCallId: "tool-1",
toolName: "read_file",
result: "file contents",
isError: false,
});
expect(result.notices).toEqual(["warn: noisy setup output"]);
});
});
describe("buildIssueChatMessages", () => {
it("orders events before comments and appends active live runs as running assistant messages", () => {
const agentMap = new Map<string, Agent>([["agent-1", createAgent("agent-1", "CodexCoder")]]);
const comments = [
createComment(),
createComment({
id: "comment-2",
authorAgentId: "agent-1",
authorUserId: null,
body: "I made the change.",
createdAt: new Date("2026-04-06T12:03:00.000Z"),
updatedAt: new Date("2026-04-06T12:03:00.000Z"),
runId: "run-1",
runAgentId: "agent-1",
}),
];
const timelineEvents: IssueTimelineEvent[] = [
{
id: "event-1",
createdAt: new Date("2026-04-06T11:59:00.000Z"),
actorType: "user",
actorId: "user-1",
statusChange: {
from: "done",
to: "todo",
},
},
];
const linkedRuns: IssueChatLinkedRun[] = [
{
runId: "run-history-1",
status: "succeeded",
agentId: "agent-1",
createdAt: new Date("2026-04-06T12:01:00.000Z"),
startedAt: new Date("2026-04-06T12:01:00.000Z"),
finishedAt: new Date("2026-04-06T12:02:00.000Z"),
},
];
const liveRuns: LiveRunForIssue[] = [
{
id: "run-live-1",
status: "running",
invocationSource: "manual",
triggerDetail: null,
startedAt: "2026-04-06T12:04:00.000Z",
finishedAt: null,
createdAt: "2026-04-06T12:04:00.000Z",
agentId: "agent-1",
agentName: "CodexCoder",
adapterType: "codex_local",
},
];
const messages = buildIssueChatMessages({
comments,
timelineEvents,
linkedRuns,
liveRuns,
transcriptsByRunId: new Map([
[
"run-live-1",
[{ kind: "assistant", ts: "2026-04-06T12:04:01.000Z", text: "Streaming reply" }],
],
]),
hasOutputForRun: () => true,
companyId: "company-1",
projectId: "project-1",
agentMap,
currentUserId: "user-1",
});
expect(messages.map((message) => `${message.role}:${message.id}`)).toEqual([
"system:activity:event-1",
"user:comment-1",
"system:run:run-history-1",
"assistant:comment-2",
"assistant:live-run:run-live-1",
]);
const liveRunMessage = messages.at(-1);
expect(liveRunMessage).toMatchObject({
role: "assistant",
status: { type: "running" },
});
expect(liveRunMessage?.content[0]).toMatchObject({
type: "text",
text: "Streaming reply",
});
});
});

View file

@ -0,0 +1,502 @@
import type {
ReasoningMessagePart,
TextMessagePart,
ThreadAssistantMessage,
ThreadMessage,
ToolCallMessagePart,
ThreadSystemMessage,
ThreadUserMessage,
} from "@assistant-ui/react";
import type { Agent, IssueComment } from "@paperclipai/shared";
import type { ActiveRunForIssue, LiveRunForIssue } from "../api/heartbeats";
import { formatAssigneeUserLabel } from "./assignees";
import type { IssueTimelineEvent } from "./issue-timeline-events";
type JsonValue = null | string | number | boolean | JsonValue[] | { [key: string]: JsonValue };
type JsonObject = { [key: string]: JsonValue };
export interface IssueChatComment extends IssueComment {
runId?: string | null;
runAgentId?: string | null;
interruptedRunId?: string | null;
clientId?: string;
clientStatus?: "pending" | "queued";
queueState?: "queued";
queueTargetRunId?: string | null;
}
export interface IssueChatLinkedRun {
runId: string;
status: string;
agentId: string;
createdAt: Date | string;
startedAt: Date | string | null;
finishedAt?: Date | string | null;
}
export interface IssueChatTranscriptEntry {
kind:
| "assistant"
| "thinking"
| "user"
| "tool_call"
| "tool_result"
| "init"
| "result"
| "stderr"
| "system"
| "stdout"
| "diff";
ts: string;
text?: string;
name?: string;
input?: unknown;
toolUseId?: string;
toolName?: string;
content?: string;
isError?: boolean;
subtype?: string;
errors?: string[];
}
type MessageWithOrder = {
createdAtMs: number;
order: number;
message: ThreadMessage;
};
function toDate(value: Date | string | null | undefined) {
return value instanceof Date ? value : new Date(value ?? Date.now());
}
function toTimestamp(value: Date | string | null | undefined) {
return toDate(value).getTime();
}
function sortByCreated<T extends { createdAt: Date | string; id: string }>(items: readonly T[]) {
return [...items].sort((a, b) => {
const diff = toTimestamp(a.createdAt) - toTimestamp(b.createdAt);
if (diff !== 0) return diff;
return a.id.localeCompare(b.id);
});
}
function normalizeJsonValue(input: unknown): JsonValue {
if (
input === null ||
typeof input === "string" ||
typeof input === "number" ||
typeof input === "boolean"
) {
return input;
}
if (Array.isArray(input)) {
return input.map((entry) => normalizeJsonValue(entry));
}
if (typeof input === "object" && input) {
const entries = Object.entries(input as Record<string, unknown>).map(([key, value]) => [
key,
normalizeJsonValue(value),
]);
return Object.fromEntries(entries) as JsonObject;
}
return String(input);
}
function normalizeToolArgs(input: unknown): JsonObject {
if (typeof input === "object" && input && !Array.isArray(input)) {
return normalizeJsonValue(input) as JsonObject;
}
if (input === undefined) return {};
return { value: normalizeJsonValue(input) };
}
function stringifyUnknown(value: unknown) {
if (typeof value === "string") return value;
if (value === null || value === undefined) return "";
try {
return JSON.stringify(value, null, 2);
} catch {
return String(value);
}
}
function createAssistantMetadata(custom: Record<string, unknown>) {
return {
unstable_state: null,
unstable_annotations: [],
unstable_data: [],
steps: [],
custom,
} as const;
}
function authorNameForComment(
comment: IssueChatComment,
agentMap?: Map<string, Agent>,
currentUserId?: string | null,
) {
if (comment.authorAgentId) {
return agentMap?.get(comment.authorAgentId)?.name ?? comment.authorAgentId.slice(0, 8);
}
return formatAssigneeUserLabel(comment.authorUserId ?? null, currentUserId) ?? "You";
}
function formatStatusLabel(status: string) {
return status.replace(/_/g, " ");
}
function createCommentMessage(args: {
comment: IssueChatComment;
agentMap?: Map<string, Agent>;
currentUserId?: string | null;
companyId?: string | null;
projectId?: string | null;
}): ThreadMessage {
const { comment, agentMap, currentUserId, companyId, projectId } = args;
const createdAt = toDate(comment.createdAt);
const authorName = authorNameForComment(comment, agentMap, currentUserId);
const custom = {
kind: "comment",
commentId: comment.id,
anchorId: `comment-${comment.id}`,
authorName,
authorAgentId: comment.authorAgentId,
authorUserId: comment.authorUserId,
companyId: companyId ?? comment.companyId,
projectId: projectId ?? null,
runId: comment.runId ?? null,
runAgentId: comment.runAgentId ?? null,
clientStatus: comment.clientStatus ?? null,
queueState: comment.queueState ?? null,
queueTargetRunId: comment.queueTargetRunId ?? null,
interruptedRunId: comment.interruptedRunId ?? null,
};
if (comment.authorAgentId) {
const message: ThreadAssistantMessage = {
id: comment.id,
role: "assistant",
createdAt,
content: [{ type: "text", text: comment.body }],
status: { type: "complete", reason: "stop" },
metadata: createAssistantMetadata(custom),
};
return message;
}
const message: ThreadUserMessage = {
id: comment.id,
role: "user",
createdAt,
content: [{ type: "text", text: comment.body }],
attachments: [],
metadata: { custom },
};
return message;
}
function createTimelineEventMessage(args: {
event: IssueTimelineEvent;
agentMap?: Map<string, Agent>;
currentUserId?: string | null;
}) {
const { event, agentMap, currentUserId } = args;
const actorName = event.actorType === "agent"
? (agentMap?.get(event.actorId)?.name ?? event.actorId.slice(0, 8))
: event.actorType === "system"
? "System"
: (formatAssigneeUserLabel(event.actorId, currentUserId) ?? "Board");
const lines: string[] = [`${actorName} updated this issue`];
if (event.statusChange) {
lines.push(
`Status: ${event.statusChange.from ?? "none"} -> ${event.statusChange.to ?? "none"}`,
);
}
if (event.assigneeChange) {
const from = event.assigneeChange.from.agentId
? (agentMap?.get(event.assigneeChange.from.agentId)?.name ?? event.assigneeChange.from.agentId.slice(0, 8))
: (formatAssigneeUserLabel(event.assigneeChange.from.userId, currentUserId) ?? "Unassigned");
const to = event.assigneeChange.to.agentId
? (agentMap?.get(event.assigneeChange.to.agentId)?.name ?? event.assigneeChange.to.agentId.slice(0, 8))
: (formatAssigneeUserLabel(event.assigneeChange.to.userId, currentUserId) ?? "Unassigned");
lines.push(`Assignee: ${from} -> ${to}`);
}
const message: ThreadSystemMessage = {
id: `activity:${event.id}`,
role: "system",
createdAt: toDate(event.createdAt),
content: [{ type: "text", text: lines.join("\n") }],
metadata: {
custom: {
kind: "event",
anchorId: `activity-${event.id}`,
eventId: event.id,
actorName,
statusChange: event.statusChange ?? null,
assigneeChange: event.assigneeChange ?? null,
},
},
};
return message;
}
function runTimestamp(run: IssueChatLinkedRun) {
return run.finishedAt ?? run.startedAt ?? run.createdAt;
}
function createHistoricalRunMessage(run: IssueChatLinkedRun, agentMap?: Map<string, Agent>) {
const agentName = agentMap?.get(run.agentId)?.name ?? run.agentId.slice(0, 8);
const message: ThreadSystemMessage = {
id: `run:${run.runId}`,
role: "system",
createdAt: toDate(runTimestamp(run)),
content: [{ type: "text", text: `${agentName} run ${run.runId.slice(0, 8)} ${formatStatusLabel(run.status)}` }],
metadata: {
custom: {
kind: "run",
anchorId: `run-${run.runId}`,
runId: run.runId,
runAgentId: run.agentId,
runAgentName: agentName,
runStatus: run.status,
},
},
};
return message;
}
function mergeAdjacentTextParts(parts: Array<TextMessagePart | ReasoningMessagePart>) {
const merged: Array<TextMessagePart | ReasoningMessagePart> = [];
for (const part of parts) {
const previous = merged.at(-1);
if (previous && previous.type === part.type && previous.parentId === part.parentId) {
merged[merged.length - 1] = {
...previous,
text: `${previous.text}${part.text}`,
};
continue;
}
merged.push(part);
}
return merged;
}
export function buildAssistantPartsFromTranscript(entries: readonly IssueChatTranscriptEntry[]) {
const textLikeParts: Array<TextMessagePart | ReasoningMessagePart> = [];
const toolParts = new Map<string, ToolCallMessagePart<JsonObject, unknown>>();
const toolOrder: string[] = [];
const notices: string[] = [];
for (const [index, entry] of entries.entries()) {
if (entry.kind === "assistant" && entry.text) {
textLikeParts.push({ type: "text", text: entry.text });
continue;
}
if (entry.kind === "thinking" && entry.text) {
textLikeParts.push({ type: "reasoning", text: entry.text });
continue;
}
if (entry.kind === "tool_call") {
const toolCallId = entry.toolUseId || `tool-${index}`;
if (!toolParts.has(toolCallId)) {
toolOrder.push(toolCallId);
}
toolParts.set(toolCallId, {
type: "tool-call",
toolCallId,
toolName: entry.name || "tool",
args: normalizeToolArgs(entry.input),
argsText: stringifyUnknown(entry.input),
});
continue;
}
if (entry.kind === "tool_result") {
const toolCallId = entry.toolUseId || `tool-result-${index}`;
const existing = toolParts.get(toolCallId);
if (!existing) {
toolOrder.push(toolCallId);
}
toolParts.set(toolCallId, {
type: "tool-call",
toolCallId,
toolName: existing?.toolName || entry.toolName || "tool",
args: existing?.args ?? {},
argsText: existing?.argsText ?? "",
result: entry.content ?? "",
isError: entry.isError === true,
});
continue;
}
if (entry.kind === "stderr" && entry.text) {
notices.push(entry.text);
continue;
}
if (entry.kind === "system" && entry.text) {
notices.push(entry.text);
continue;
}
if (entry.kind === "result") {
if (entry.isError && entry.errors?.length) {
notices.push(...entry.errors);
} else if (entry.text) {
notices.push(entry.text);
}
}
}
return {
parts: [
...mergeAdjacentTextParts(textLikeParts),
...toolOrder
.map((toolCallId) => toolParts.get(toolCallId))
.filter((part): part is ToolCallMessagePart<JsonObject, unknown> => Boolean(part)),
],
notices,
};
}
function normalizeLiveRuns(
liveRuns: readonly LiveRunForIssue[],
activeRun: ActiveRunForIssue | null | undefined,
issueId?: string,
) {
const deduped = new Map<string, LiveRunForIssue>();
for (const run of liveRuns) {
deduped.set(run.id, run);
}
if (activeRun) {
deduped.set(activeRun.id, {
id: activeRun.id,
status: activeRun.status,
invocationSource: activeRun.invocationSource,
triggerDetail: activeRun.triggerDetail,
startedAt: activeRun.startedAt ? toDate(activeRun.startedAt).toISOString() : null,
finishedAt: activeRun.finishedAt ? toDate(activeRun.finishedAt).toISOString() : null,
createdAt: toDate(activeRun.createdAt).toISOString(),
agentId: activeRun.agentId,
agentName: activeRun.agentName,
adapterType: activeRun.adapterType,
issueId,
});
}
return [...deduped.values()].sort((a, b) => toTimestamp(a.createdAt) - toTimestamp(b.createdAt));
}
function createLiveRunMessage(args: {
run: LiveRunForIssue;
transcript: readonly IssueChatTranscriptEntry[];
hasOutput: boolean;
}) {
const { run, transcript, hasOutput } = args;
const { parts, notices } = buildAssistantPartsFromTranscript(transcript);
const waitingText =
run.status === "queued"
? "Queued..."
: hasOutput
? ""
: "Working...";
const content = parts.length > 0
? parts
: waitingText
? [{ type: "text", text: waitingText } satisfies TextMessagePart]
: [];
const message: ThreadAssistantMessage = {
id: `live-run:${run.id}`,
role: "assistant",
createdAt: toDate(run.startedAt ?? run.createdAt),
content,
status: { type: "running" },
metadata: createAssistantMetadata({
kind: "live-run",
runId: run.id,
runAgentId: run.agentId,
runAgentName: run.agentName,
runStatus: run.status,
adapterType: run.adapterType,
notices,
waitingText,
}),
};
return message;
}
export function buildIssueChatMessages(args: {
comments: readonly IssueChatComment[];
timelineEvents: readonly IssueTimelineEvent[];
linkedRuns: readonly IssueChatLinkedRun[];
liveRuns: readonly LiveRunForIssue[];
activeRun?: ActiveRunForIssue | null;
transcriptsByRunId?: ReadonlyMap<string, readonly IssueChatTranscriptEntry[]>;
hasOutputForRun?: (runId: string) => boolean;
issueId?: string;
companyId?: string | null;
projectId?: string | null;
agentMap?: Map<string, Agent>;
currentUserId?: string | null;
}) {
const {
comments,
timelineEvents,
linkedRuns,
liveRuns,
activeRun,
transcriptsByRunId,
hasOutputForRun,
issueId,
companyId,
projectId,
agentMap,
currentUserId,
} = args;
const orderedMessages: MessageWithOrder[] = [];
for (const comment of sortByCreated(comments)) {
orderedMessages.push({
createdAtMs: toTimestamp(comment.createdAt),
order: 1,
message: createCommentMessage({ comment, agentMap, currentUserId, companyId, projectId }),
});
}
for (const event of sortByCreated(timelineEvents)) {
orderedMessages.push({
createdAtMs: toTimestamp(event.createdAt),
order: 0,
message: createTimelineEventMessage({ event, agentMap, currentUserId }),
});
}
for (const run of [...linkedRuns].sort((a, b) => toTimestamp(runTimestamp(a)) - toTimestamp(runTimestamp(b)))) {
orderedMessages.push({
createdAtMs: toTimestamp(runTimestamp(run)),
order: 2,
message: createHistoricalRunMessage(run, agentMap),
});
}
for (const run of normalizeLiveRuns(liveRuns, activeRun, issueId)) {
orderedMessages.push({
createdAtMs: toTimestamp(run.startedAt ?? run.createdAt),
order: 3,
message: createLiveRunMessage({
run,
transcript: transcriptsByRunId?.get(run.id) ?? [],
hasOutput: hasOutputForRun?.(run.id) ?? false,
}),
});
}
return orderedMessages
.sort((a, b) => {
if (a.createdAtMs !== b.createdAtMs) return a.createdAtMs - b.createdAtMs;
if (a.order !== b.order) return a.order - b.order;
return a.message.id.localeCompare(b.message.id);
})
.map((entry) => entry.message);
}