@tstax/coding-tab 0.2.1 → 0.2.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/browser.js +51 -44
- package/dist/browser.js.map +1 -1
- package/dist/server.cjs +26 -5
- package/dist/server.cjs.map +1 -1
- package/dist/server.d.cts +18 -0
- package/dist/server.d.ts +18 -0
- package/dist/server.js +26 -5
- package/dist/server.js.map +1 -1
- package/dist/style.css +22 -3
- package/package.json +1 -1
package/dist/browser.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/client/tab.ts","../src/client/markdown.ts"],"sourcesContent":["import { escapeHtml, renderMarkdown } from \"./markdown.js\";\nimport type {\n ChatListItem,\n ChatMode,\n FullChat,\n MeResponse,\n ModelChoice,\n ModelOption,\n PrInfo,\n StoredChat,\n StoredTurn,\n StreamEvent,\n TimelineEvent,\n TurnStatus,\n} from \"../shared/types.js\";\n\ninterface MountOptions {\n apiBase: string;\n defaultRepo?: string;\n defaultRef?: string;\n defaultMode?: ChatMode;\n defaultModel?: ModelChoice;\n}\n\ninterface MountHandle {\n destroy(): void;\n}\n\n/**\n * Visible state of a \"Merge & Redeploy\" button. Stored per PR-url so the\n * feedback survives re-renders (which happen frequently during streaming) and\n * so multiple PRs in the same chat can have independent states.\n */\ntype MergeState =\n | { state: \"idle\" }\n | { state: \"loading\" }\n | { state: \"success\"; sha: string }\n | { state: \"error\"; message: string };\n\n/** Client-side ephemeral wrapper around a `StoredChat`. */\ninterface ChatThread {\n meta: ChatListItem;\n /** Loaded turns (after `/chats/:id` fetch). Undefined means not yet hydrated. */\n turns?: StoredTurn[];\n loaded: boolean;\n isStreaming: boolean;\n activeRunId: string | null;\n abort: AbortController | null;\n}\n\nconst ICON_GITHUB = `<svg width=\"18\" height=\"18\" viewBox=\"0 0 16 16\" fill=\"currentColor\"><path d=\"M8 0C3.58 0 0 3.58 0 8a8 8 0 005.47 7.59c.4.07.55-.17.55-.38v-1.34c-2.23.48-2.7-1.07-2.7-1.07-.36-.92-.89-1.16-.89-1.16-.73-.5.05-.49.05-.49.81.06 1.23.83 1.23.83.72 1.23 1.88.87 2.34.66.07-.52.28-.87.5-1.07-1.78-.2-3.65-.89-3.65-3.95 0-.87.31-1.59.82-2.15-.08-.2-.36-1.02.08-2.13 0 0 .67-.21 2.2.82a7.65 7.65 0 014 0c1.53-1.04 2.2-.82 2.2-.82.44 1.11.16 1.93.08 2.13.51.56.82 1.28.82 2.15 0 3.07-1.87 3.75-3.65 3.95.29.25.54.74.54 1.49v2.21c0 .21.15.46.55.38A8 8 0 0016 8c0-4.42-3.58-8-8-8z\"/></svg>`;\n\nconst ICON_PLUS = `<svg width=\"14\" height=\"14\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\"><path d=\"M8 3v10M3 8h10\"/></svg>`;\nconst ICON_TRASH = `<svg width=\"13\" height=\"13\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.4\" stroke-linecap=\"round\"><path d=\"M3 4h10M6 4V2.5a.5.5 0 01.5-.5h3a.5.5 0 01.5.5V4M5 4l.7 9.1a.9.9 0 00.9.9h2.8a.9.9 0 00.9-.9L11 4\"/></svg>`;\nconst ICON_PENCIL = `<svg width=\"13\" height=\"13\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.4\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M11 2.5l2.5 2.5L6 12.5 3 13l.5-3z\"/></svg>`;\nconst ICON_MENU = `<svg width=\"18\" height=\"18\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.6\" stroke-linecap=\"round\"><path d=\"M2.5 4h11M2.5 8h11M2.5 12h11\"/></svg>`;\nconst ICON_CHEVRON = `<svg class=\"ct-chevron\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M3 2l4 3-4 3\"/></svg>`;\n\nfunction escapeAttr(s: string): string {\n return escapeHtml(s);\n}\n\nfunction repoSlug(url: string): string {\n const m = url.match(/github\\.com[/:]([^/]+)\\/([^/]+?)(?:\\.git)?\\/?$/);\n return m ? `${m[1]}/${m[2]}` : url;\n}\n\nfunction cryptoRandomId(): string {\n return typeof crypto !== \"undefined\" && \"randomUUID\" in crypto\n ? crypto.randomUUID()\n : Math.random().toString(36).slice(2);\n}\n\nfunction relativeTime(ts: number): string {\n const diff = Date.now() - ts;\n if (diff < 60_000) return \"just now\";\n if (diff < 3_600_000) return `${Math.round(diff / 60_000)}m`;\n if (diff < 86_400_000) return `${Math.round(diff / 3_600_000)}h`;\n if (diff < 604_800_000) return `${Math.round(diff / 86_400_000)}d`;\n return new Date(ts).toLocaleDateString();\n}\n\nexport function mountCodingTab(el: HTMLElement, options: MountOptions): MountHandle {\n const apiBase = options.apiBase.replace(/\\/$/, \"\");\n const root = document.createElement(\"div\");\n root.className = \"coding-tab\";\n el.innerHTML = \"\";\n el.appendChild(root);\n\n let stylesheetInjected = false;\n function injectStylesheet() {\n if (stylesheetInjected) return;\n stylesheetInjected = true;\n if (document.querySelector(`link[data-coding-tab=\"style\"]`)) return;\n const link = document.createElement(\"link\");\n link.rel = \"stylesheet\";\n link.href = `${apiBase}/style.css`;\n link.dataset.codingTab = \"style\";\n document.head.appendChild(link);\n }\n injectStylesheet();\n\n const state: {\n me: MeResponse | null;\n models: ModelOption[];\n mode: ChatMode;\n model: ModelChoice;\n repoUrl: string;\n repoLocked: boolean;\n chats: ChatThread[];\n activeChatId: string | null;\n sidebarOpen: boolean;\n /** Tracks expanded tool events across re-renders (keyed by `event.id`). */\n expandedTools: Set<string>;\n /** Per-PR merge state so the button shows loading/success/error feedback. */\n mergeStates: Map<string, MergeState>;\n } = {\n me: null,\n models: [],\n mode: options.defaultMode ?? \"plan\",\n model: options.defaultModel ?? \"sonnet\",\n repoUrl: options.defaultRepo ?? \"\",\n repoLocked: false,\n chats: [],\n activeChatId: null,\n sidebarOpen: window.innerWidth >= 720,\n expandedTools: new Set(),\n mergeStates: new Map(),\n };\n\n function activeChat(): ChatThread | null {\n return state.chats.find((c) => c.meta.id === state.activeChatId) ?? null;\n }\n\n async function api<T = unknown>(path: string, init?: RequestInit): Promise<T> {\n const resp = await fetch(`${apiBase}${path}`, {\n credentials: \"include\",\n headers: { \"Content-Type\": \"application/json\", ...(init?.headers ?? {}) },\n ...init,\n });\n if (!resp.ok)\n throw Object.assign(new Error(`${resp.status} ${resp.statusText}`), { status: resp.status });\n return (await resp.json()) as T;\n }\n\n // ───────── chat lifecycle ─────────\n\n async function loadChatList(): Promise<void> {\n try {\n const { chats } = await api<{ chats: ChatListItem[] }>(\"/chats\");\n // Preserve runtime state for chats we already had.\n const existing = new Map(state.chats.map((c) => [c.meta.id, c]));\n state.chats = chats.map((meta) => {\n const prev = existing.get(meta.id);\n return prev\n ? { ...prev, meta }\n : {\n meta,\n loaded: false,\n isStreaming: false,\n activeRunId: null,\n abort: null,\n };\n });\n } catch (err) {\n console.error(\"[coding-tab] /chats list failed\", err);\n state.chats = [];\n }\n }\n\n async function ensureChatLoaded(chatId: string): Promise<ChatThread | null> {\n const t = state.chats.find((c) => c.meta.id === chatId);\n if (!t) return null;\n if (t.loaded) return t;\n try {\n const { chat } = await api<{ chat: FullChat }>(`/chats/${chatId}`);\n t.turns = chat.turns;\n t.loaded = true;\n t.meta = {\n id: chat.id,\n title: chat.title,\n mode: chat.mode,\n model: chat.model,\n createdAt: chat.createdAt,\n updatedAt: chat.updatedAt,\n };\n // Sync composer state to whatever this chat last used.\n state.mode = chat.mode;\n state.model = chat.model;\n return t;\n } catch (err) {\n console.error(`[coding-tab] /chats/${chatId} load failed`, err);\n return null;\n }\n }\n\n async function createChat(): Promise<ChatThread | null> {\n try {\n const { chat } = await api<{ chat: StoredChat }>(\"/chats\", {\n method: \"POST\",\n body: JSON.stringify({\n mode: state.mode,\n model: state.model,\n repoUrl: state.repoLocked ? undefined : state.repoUrl || undefined,\n }),\n });\n const thread: ChatThread = {\n meta: {\n id: chat.id,\n title: chat.title,\n mode: chat.mode,\n model: chat.model,\n createdAt: chat.createdAt,\n updatedAt: chat.updatedAt,\n },\n turns: [],\n loaded: true,\n isStreaming: false,\n activeRunId: null,\n abort: null,\n };\n state.chats.unshift(thread);\n state.activeChatId = thread.meta.id;\n return thread;\n } catch (err) {\n console.error(\"[coding-tab] create chat failed\", err);\n return null;\n }\n }\n\n async function switchChat(chatId: string): Promise<void> {\n state.activeChatId = chatId;\n if (window.innerWidth < 720) state.sidebarOpen = false;\n render();\n await ensureChatLoaded(chatId);\n render();\n }\n\n async function deleteChat(chatId: string): Promise<void> {\n const thread = state.chats.find((c) => c.meta.id === chatId);\n if (!thread) return;\n if (!confirm(`Delete chat \"${thread.meta.title}\"? This cannot be undone.`)) return;\n thread.abort?.abort();\n try {\n await api(`/chats/${chatId}`, { method: \"DELETE\" });\n } catch (err) {\n console.error(`[coding-tab] delete chat ${chatId} failed`, err);\n return;\n }\n state.chats = state.chats.filter((c) => c.meta.id !== chatId);\n if (state.activeChatId === chatId) {\n state.activeChatId = state.chats[0]?.meta.id ?? null;\n }\n render();\n }\n\n async function renameChat(chatId: string): Promise<void> {\n const thread = state.chats.find((c) => c.meta.id === chatId);\n if (!thread) return;\n const next = prompt(\"Rename chat\", thread.meta.title);\n if (next === null) return;\n const trimmed = next.trim();\n if (!trimmed || trimmed === thread.meta.title) return;\n try {\n const { chat } = await api<{ chat: StoredChat }>(`/chats/${chatId}`, {\n method: \"PATCH\",\n body: JSON.stringify({ title: trimmed }),\n });\n thread.meta = { ...thread.meta, title: chat.title, updatedAt: chat.updatedAt };\n render();\n } catch (err) {\n console.error(`[coding-tab] rename chat ${chatId} failed`, err);\n }\n }\n\n // ───────── render ─────────\n\n function render() {\n if (!state.me) {\n root.innerHTML = `\n <div class=\"coding-tab__signin\">\n <h2>Coding Tab</h2>\n <p>Sign in with GitHub to start a coding session against your repo. Your token is used to clone, push, and open pull requests on your behalf.</p>\n <a href=\"${apiBase}/auth/login\">${ICON_GITHUB}<span>Sign in with GitHub</span></a>\n </div>\n `;\n return;\n }\n\n const headerHtml = `\n <div class=\"coding-tab__header\">\n <button class=\"coding-tab__hamburger\" data-role=\"toggle-sidebar\" aria-label=\"Toggle chat list\">${ICON_MENU}</button>\n <select data-role=\"model\" title=\"Model\">\n ${state.models\n .map(\n (m) =>\n `<option value=\"${m.choice}\" ${state.model === m.choice ? \"selected\" : \"\"}>${escapeHtml(m.displayName)}</option>`,\n )\n .join(\"\")}\n </select>\n <div class=\"coding-tab__mode\">\n <button data-mode=\"plan\" class=\"${state.mode === \"plan\" ? \"is-active\" : \"\"}\">Plan</button>\n <button data-mode=\"agent\" class=\"${state.mode === \"agent\" ? \"is-active\" : \"\"}\">Agent</button>\n </div>\n ${\n state.repoLocked && state.repoUrl\n ? `<a class=\"coding-tab__repo-locked\" href=\"${escapeAttr(state.repoUrl)}\" target=\"_blank\" rel=\"noreferrer\" title=\"Locked to this app's repo (set by the server)\">${ICON_GITHUB}<span>${escapeHtml(repoSlug(state.repoUrl))}</span></a>`\n : `<input data-role=\"repo\" placeholder=\"https://github.com/org/repo\" value=\"${escapeAttr(state.repoUrl)}\" style=\"flex:1;min-width:200px\" />`\n }\n <div class=\"coding-tab__user\">\n ${state.me.avatarUrl ? `<img src=\"${state.me.avatarUrl}\" alt=\"\" />` : \"\"}\n <span class=\"coding-tab__user-name\">@${escapeHtml(state.me.githubLogin)}</span>\n <button data-role=\"logout\">Sign out</button>\n </div>\n </div>\n `;\n\n const sidebarHtml = `\n <aside class=\"coding-tab__sidebar ${state.sidebarOpen ? \"is-open\" : \"\"}\">\n <button class=\"coding-tab__btn primary coding-tab__new-chat\" data-role=\"new-chat\">${ICON_PLUS}<span>New chat</span></button>\n <div class=\"coding-tab__chat-list\">\n ${\n state.chats.length === 0\n ? `<div class=\"coding-tab__chat-empty\">No chats yet — start one with the button above.</div>`\n : state.chats.map((c) => renderChatRow(c)).join(\"\")\n }\n </div>\n </aside>\n `;\n\n const chat = activeChat();\n const paneHtml = `\n <main class=\"coding-tab__pane\">\n ${chat ? renderChatPane(chat) : renderEmptyPane()}\n </main>\n `;\n\n root.innerHTML = `\n ${headerHtml}\n <div class=\"coding-tab__body\">\n ${sidebarHtml}\n ${paneHtml}\n </div>\n `;\n\n const thread = root.querySelector(\"[data-role=thread]\") as HTMLElement | null;\n if (thread) thread.scrollTop = thread.scrollHeight;\n\n bindEvents();\n }\n\n function renderChatRow(c: ChatThread): string {\n const active = c.meta.id === state.activeChatId ? \" is-active\" : \"\";\n const streaming = c.isStreaming ? `<span class=\"coding-tab__chat-row-dot\" title=\"Streaming\"></span>` : \"\";\n return `\n <div class=\"coding-tab__chat-row${active}\" data-chat-id=\"${c.meta.id}\">\n <button class=\"coding-tab__chat-row-main\" data-role=\"select-chat\" data-chat-id=\"${c.meta.id}\">\n ${streaming}\n <span class=\"coding-tab__chat-row-title\">${escapeHtml(c.meta.title)}</span>\n <span class=\"coding-tab__chat-row-meta\">${c.meta.mode === \"plan\" ? \"Plan\" : \"Agent\"} · ${relativeTime(c.meta.updatedAt)}</span>\n </button>\n <div class=\"coding-tab__chat-row-actions\">\n <button data-role=\"rename-chat\" data-chat-id=\"${c.meta.id}\" title=\"Rename\">${ICON_PENCIL}</button>\n <button data-role=\"delete-chat\" data-chat-id=\"${c.meta.id}\" title=\"Delete\">${ICON_TRASH}</button>\n </div>\n </div>\n `;\n }\n\n function renderEmptyPane(): string {\n return `\n <div class=\"coding-tab__pane-empty\">\n <h3>Start a new chat</h3>\n <p>Pick a mode (Plan or Agent), choose a model, and click <strong>New chat</strong> to begin. Each chat keeps its own context.</p>\n <button class=\"coding-tab__btn primary\" data-role=\"new-chat\">${ICON_PLUS}<span>New chat</span></button>\n </div>\n `;\n }\n\n function renderChatPane(chat: ChatThread): string {\n const turns = chat.turns ?? [];\n const threadHtml =\n !chat.loaded\n ? `<div class=\"coding-tab__notice info\">Loading…</div>`\n : turns.length === 0\n ? `<div class=\"coding-tab__notice info\">Type a message below to kick off this chat.</div>`\n : turns.map((t) => renderTurn(t)).join(\"\");\n\n const placeholder =\n turns.length === 0\n ? `Ask anything (e.g. \"add dark mode that follows the OS setting\")`\n : state.mode === \"plan\"\n ? \"Refine the plan, or ask another planning question\"\n : \"Send another instruction\";\n\n return `\n <div class=\"coding-tab__thread\" data-role=\"thread\">${threadHtml}</div>\n <div class=\"coding-tab__composer\">\n <textarea data-role=\"prompt\" placeholder=\"${escapeAttr(placeholder)}\" rows=\"2\"></textarea>\n <button class=\"coding-tab__btn primary\" data-role=\"send\" ${chat.isStreaming ? \"disabled\" : \"\"}>${chat.isStreaming ? \"Working…\" : \"Send\"}</button>\n ${chat.isStreaming ? `<button class=\"coding-tab__btn danger\" data-role=\"cancel\">Stop</button>` : \"\"}\n </div>\n `;\n }\n\n function renderTurn(turn: StoredTurn): string {\n if (turn.role === \"user\") {\n const text = turn.prompt ?? extractText(turn);\n return `<div class=\"coding-tab__msg user\">\n <div class=\"coding-tab__msg-role\">You · ${turn.isPlan ? \"Plan\" : \"Agent\"}</div>\n <div class=\"coding-tab__msg-body\">${escapeHtml(text)}</div>\n </div>`;\n }\n\n const eventsHtml =\n turn.events.length === 0 && turn.status === \"running\"\n ? `<div class=\"coding-tab__msg-pending\">Working…</div>`\n : turn.events.map((evt) => renderEvent(evt, turn.isPlan)).join(\"\");\n\n const statusBadge = turn.status && turn.status !== \"finished\"\n ? `<div class=\"coding-tab__status\">${escapeHtml(turn.status)}</div>`\n : \"\";\n\n const planActions =\n turn.showExecute && !activeChat()?.isStreaming\n ? `<div class=\"coding-tab__plan-actions\">\n <button class=\"coding-tab__btn primary\" data-role=\"execute\" data-turn=\"${turn.id}\">Execute plan</button>\n </div>`\n : \"\";\n\n const prHtml = turn.pr ? renderPr(turn.pr) : \"\";\n\n return `<div class=\"coding-tab__msg assistant\">\n <div class=\"coding-tab__msg-role\">Coding Tab · ${turn.isPlan ? \"Plan\" : \"Agent\"} ${statusBadge}</div>\n <div class=\"coding-tab__msg-body\">${eventsHtml}${planActions}${prHtml}</div>\n </div>`;\n }\n\n function renderEvent(evt: TimelineEvent, isPlan: boolean): string {\n if (evt.kind === \"text\") {\n const text = evt.text.trim();\n if (!text) return \"\";\n const html = isPlan ? renderMarkdown(text) : renderInlineText(text);\n return `<div class=\"coding-tab__md\">${html}</div>`;\n }\n return renderToolEvent(evt);\n }\n\n function renderInlineText(text: string): string {\n // Lightweight rendering for Agent-mode messages: keep paragraph breaks,\n // inline `code` and **bold**, but skip headings/lists so the model's casual\n // streaming output doesn't get over-formatted.\n return escapeHtml(text)\n .split(/\\n\\s*\\n/)\n .map((para) =>\n para\n .replace(/`([^`\\n]+)`/g, \"<code>$1</code>\")\n .replace(/\\*\\*([^*\\n]+)\\*\\*/g, \"<strong>$1</strong>\")\n .replace(/\\n/g, \"<br />\"),\n )\n .map((p) => `<p>${p}</p>`)\n .join(\"\");\n }\n\n function renderToolEvent(\n evt: Extract<TimelineEvent, { kind: \"tool\" }>,\n ): string {\n const summary = summarizeTool(evt.name, evt.args);\n const expanded = state.expandedTools.has(evt.id);\n const detailHtml = expanded ? renderToolDetail(evt) : \"\";\n return `\n <div class=\"coding-tab__tool\" data-status=\"${evt.status}\" data-tool-id=\"${evt.id}\">\n <button class=\"coding-tab__tool-header\" data-role=\"toggle-tool\" data-tool-id=\"${evt.id}\" aria-expanded=\"${expanded}\">\n <span class=\"coding-tab__tool-dot\"></span>\n ${ICON_CHEVRON}\n <span class=\"coding-tab__tool-name\">${escapeHtml(evt.name)}</span>\n ${summary ? `<span class=\"coding-tab__tool-summary\">${escapeHtml(summary)}</span>` : \"\"}\n </button>\n ${detailHtml}\n </div>\n `;\n }\n\n function summarizeTool(name: string, args: unknown): string {\n if (!args || typeof args !== \"object\") return \"\";\n const a = args as Record<string, unknown>;\n const lower = name.toLowerCase();\n if (lower.includes(\"grep\")) {\n const pattern = typeof a.pattern === \"string\" ? a.pattern : typeof a.query === \"string\" ? a.query : \"\";\n return pattern ? `\"${pattern}\"` : \"\";\n }\n if (lower.includes(\"read\") && typeof a.path === \"string\") {\n const lines =\n typeof a.offset === \"number\" || typeof a.limit === \"number\"\n ? `:${a.offset ?? \"\"}-${(Number(a.offset ?? 0) + Number(a.limit ?? 0)) || \"\"}`\n : \"\";\n return `${a.path}${lines}`;\n }\n if ((lower.includes(\"glob\") || lower.includes(\"file_search\")) && typeof a.glob_pattern === \"string\") {\n return a.glob_pattern as string;\n }\n if (lower === \"shell\" && typeof a.command === \"string\") {\n return (a.command as string).slice(0, 80);\n }\n if (lower === \"task\" && typeof a.description === \"string\") {\n return a.description as string;\n }\n if (lower.includes(\"write\") && typeof a.path === \"string\") {\n return a.path as string;\n }\n if (lower.includes(\"edit\") && typeof a.path === \"string\") {\n return a.path as string;\n }\n if (lower.includes(\"delete\") && typeof a.path === \"string\") {\n return a.path as string;\n }\n return \"\";\n }\n\n function renderToolDetail(evt: Extract<TimelineEvent, { kind: \"tool\" }>): string {\n const argsBlock =\n evt.args !== undefined && evt.args !== null\n ? `<div class=\"coding-tab__tool-section\"><div class=\"coding-tab__tool-label\">Args</div><pre>${escapeHtml(formatBlob(evt.args))}</pre></div>`\n : \"\";\n const resultBlock =\n evt.result !== undefined && evt.result !== null\n ? `<div class=\"coding-tab__tool-section\"><div class=\"coding-tab__tool-label\">Result</div><pre>${escapeHtml(formatBlob(evt.result))}</pre></div>`\n : evt.status === \"running\"\n ? `<div class=\"coding-tab__tool-section\"><div class=\"coding-tab__tool-label\">Result</div><pre class=\"coding-tab__tool-pending\">…running</pre></div>`\n : \"\";\n return `<div class=\"coding-tab__tool-detail\">${argsBlock}${resultBlock}</div>`;\n }\n\n function formatBlob(value: unknown): string {\n if (typeof value === \"string\") return value;\n try {\n return JSON.stringify(value, null, 2);\n } catch {\n return String(value);\n }\n }\n\n function renderPr(pr: PrInfo): string {\n const merge = state.mergeStates.get(pr.url) ?? { state: \"idle\" as const };\n let mergeButton: string;\n let statusLine = \"\";\n switch (merge.state) {\n case \"loading\":\n mergeButton = `<button class=\"coding-tab__btn primary\" data-state=\"loading\" disabled>Merging…</button>`;\n statusLine = `<div class=\"coding-tab__pr-status\">Merging this PR and triggering Railway redeploy…</div>`;\n break;\n case \"success\":\n mergeButton = `<button class=\"coding-tab__btn success\" disabled>Merged ✓</button>`;\n statusLine = `<div class=\"coding-tab__pr-status is-success\">Merged commit ${escapeHtml(merge.sha.slice(0, 7))}. Railway should redeploy on the next push hook.</div>`;\n break;\n case \"error\":\n mergeButton = `<button class=\"coding-tab__btn primary\" data-role=\"merge\" data-pr=\"${escapeAttr(pr.url)}\">Try merge again</button>`;\n statusLine = `<div class=\"coding-tab__pr-status is-error\">Merge failed: ${escapeHtml(merge.message)}</div>`;\n break;\n default:\n mergeButton = `<button class=\"coding-tab__btn primary\" data-role=\"merge\" data-pr=\"${escapeAttr(pr.url)}\">Merge & Redeploy</button>`;\n }\n return `<div class=\"coding-tab__pr\">\n <div class=\"coding-tab__pr-title\">PR #${pr.number} opened in ${escapeHtml(pr.owner)}/${escapeHtml(pr.repo)}</div>\n ${pr.title ? `<div>${escapeHtml(pr.title)}</div>` : \"\"}\n <div class=\"coding-tab__pr-actions\">\n <a class=\"coding-tab__btn\" href=\"${escapeAttr(pr.url)}\" target=\"_blank\" rel=\"noreferrer\">Open in GitHub</a>\n ${mergeButton}\n </div>\n ${statusLine}\n </div>`;\n }\n\n function extractText(turn: StoredTurn): string {\n return turn.events\n .filter((e): e is Extract<TimelineEvent, { kind: \"text\" }> => e.kind === \"text\")\n .map((e) => e.text)\n .join(\"\\n\\n\");\n }\n\n // ───────── events ─────────\n\n function bindEvents() {\n root.querySelector(`[data-role=\"model\"]`)?.addEventListener(\"change\", (e) => {\n state.model = (e.target as HTMLSelectElement).value as ModelChoice;\n });\n root.querySelectorAll(`[data-mode]`).forEach((b) => {\n b.addEventListener(\"click\", () => {\n state.mode = (b as HTMLElement).dataset.mode as ChatMode;\n render();\n });\n });\n root.querySelector(`[data-role=\"repo\"]`)?.addEventListener(\"change\", (e) => {\n state.repoUrl = (e.target as HTMLInputElement).value.trim();\n });\n root.querySelector(`[data-role=\"logout\"]`)?.addEventListener(\"click\", async () => {\n await fetch(`${apiBase}/auth/logout`, { method: \"POST\", credentials: \"include\" }).catch(() => {});\n for (const c of state.chats) c.abort?.abort();\n state.me = null;\n state.chats = [];\n state.activeChatId = null;\n render();\n });\n root.querySelector(`[data-role=\"toggle-sidebar\"]`)?.addEventListener(\"click\", () => {\n state.sidebarOpen = !state.sidebarOpen;\n render();\n });\n\n root.querySelectorAll(`[data-role=\"new-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", async () => {\n const t = await createChat();\n if (t && window.innerWidth < 720) state.sidebarOpen = false;\n render();\n }),\n );\n root.querySelectorAll(`[data-role=\"select-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", () => {\n const id = (b as HTMLElement).dataset.chatId!;\n switchChat(id);\n }),\n );\n root.querySelectorAll(`[data-role=\"rename-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", (e) => {\n e.stopPropagation();\n renameChat((b as HTMLElement).dataset.chatId!);\n }),\n );\n root.querySelectorAll(`[data-role=\"delete-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", (e) => {\n e.stopPropagation();\n deleteChat((b as HTMLElement).dataset.chatId!);\n }),\n );\n\n root.querySelector(`[data-role=\"send\"]`)?.addEventListener(\"click\", () => onSend());\n root.querySelector(`[data-role=\"prompt\"]`)?.addEventListener(\"keydown\", (e: Event) => {\n const ev = e as KeyboardEvent;\n if (ev.key === \"Enter\" && (ev.metaKey || ev.ctrlKey)) {\n ev.preventDefault();\n onSend();\n }\n });\n root.querySelector(`[data-role=\"cancel\"]`)?.addEventListener(\"click\", () => onCancel());\n root.querySelectorAll(`[data-role=\"execute\"]`).forEach((b) =>\n b.addEventListener(\"click\", () => onExecute()),\n );\n root.querySelectorAll(`[data-role=\"merge\"]`).forEach((b) => {\n b.addEventListener(\"click\", () => onMerge((b as HTMLElement).dataset.pr!));\n });\n\n root.querySelectorAll(`[data-role=\"toggle-tool\"]`).forEach((b) =>\n b.addEventListener(\"click\", () => {\n const id = (b as HTMLElement).dataset.toolId!;\n if (state.expandedTools.has(id)) state.expandedTools.delete(id);\n else state.expandedTools.add(id);\n render();\n }),\n );\n }\n\n // ───────── send / execute / cancel / merge ─────────\n\n async function onSend() {\n let chat = activeChat();\n if (!chat) {\n chat = await createChat();\n if (!chat) return;\n }\n if (chat.isStreaming) return;\n\n const promptEl = root.querySelector(`[data-role=\"prompt\"]`) as HTMLTextAreaElement | null;\n const prompt = promptEl?.value.trim() ?? \"\";\n if (!prompt) return;\n if (promptEl) promptEl.value = \"\";\n\n chat.turns = chat.turns ?? [];\n const userTurn: StoredTurn = {\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"user\",\n isPlan: state.mode === \"plan\",\n status: \"finished\",\n events: [{ kind: \"text\", id: cryptoRandomId(), text: prompt }],\n prompt,\n createdAt: Date.now(),\n };\n chat.turns.push(userTurn);\n\n const assistantTurn: StoredTurn = {\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"assistant\",\n isPlan: state.mode === \"plan\",\n status: \"running\",\n events: [],\n createdAt: Date.now(),\n };\n chat.turns.push(assistantTurn);\n chat.isStreaming = true;\n if (chat.meta.title === \"New chat\") {\n chat.meta = { ...chat.meta, title: prompt.length > 60 ? `${prompt.slice(0, 57)}…` : prompt };\n }\n chat.meta = { ...chat.meta, mode: state.mode, model: state.model, updatedAt: Date.now() };\n render();\n\n try {\n await streamSse(\n \"/agent/send\",\n {\n chatId: chat.meta.id,\n prompt,\n mode: state.mode,\n },\n chat,\n assistantTurn,\n );\n } catch (err) {\n if ((err as Error).name !== \"AbortError\") {\n appendErrorToTurn(assistantTurn, err);\n }\n assistantTurn.status = \"error\";\n } finally {\n chat.isStreaming = false;\n chat.activeRunId = null;\n chat.abort = null;\n render();\n }\n }\n\n async function onExecute() {\n const chat = activeChat();\n if (!chat || chat.isStreaming) return;\n chat.turns = chat.turns ?? [];\n const turn: StoredTurn = {\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"assistant\",\n isPlan: false,\n status: \"running\",\n events: [],\n createdAt: Date.now(),\n };\n chat.turns.push(turn);\n chat.isStreaming = true;\n render();\n try {\n await streamSse(\"/agent/execute\", { chatId: chat.meta.id }, chat, turn);\n } catch (err) {\n if ((err as Error).name !== \"AbortError\") {\n appendErrorToTurn(turn, err);\n }\n turn.status = \"error\";\n } finally {\n chat.isStreaming = false;\n chat.activeRunId = null;\n chat.abort = null;\n render();\n }\n }\n\n async function onCancel() {\n const chat = activeChat();\n if (!chat) return;\n if (chat.activeRunId) {\n try {\n await api(\"/agent/cancel\", {\n method: \"POST\",\n body: JSON.stringify({ chatId: chat.meta.id, runId: chat.activeRunId }),\n });\n } catch (err) {\n console.error(\"[coding-tab] cancel failed\", err);\n }\n }\n chat.abort?.abort();\n }\n\n async function onMerge(prUrl: string) {\n const chat = activeChat();\n if (!chat) return;\n if (state.mergeStates.get(prUrl)?.state === \"loading\") return;\n state.mergeStates.set(prUrl, { state: \"loading\" });\n render();\n\n chat.turns = chat.turns ?? [];\n try {\n const result = await api<{ sha: string; merged: boolean }>(\"/pr/merge\", {\n method: \"POST\",\n body: JSON.stringify({ prUrl, mergeMethod: \"squash\" }),\n });\n if (result.merged) {\n state.mergeStates.set(prUrl, { state: \"success\", sha: result.sha });\n chat.turns.push({\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"assistant\",\n isPlan: false,\n status: \"finished\",\n events: [\n {\n kind: \"text\",\n id: cryptoRandomId(),\n text: `Merged commit \\`${result.sha.slice(0, 7)}\\` into the default branch. Railway should redeploy on the next push hook.`,\n },\n ],\n createdAt: Date.now(),\n });\n } else {\n state.mergeStates.set(prUrl, {\n state: \"error\",\n message: \"GitHub returned merged=false\",\n });\n }\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err);\n state.mergeStates.set(prUrl, { state: \"error\", message });\n } finally {\n render();\n }\n }\n\n function appendErrorToTurn(turn: StoredTurn, err: unknown): void {\n const message = err instanceof Error ? err.message : String(err);\n turn.events.push({\n kind: \"text\",\n id: cryptoRandomId(),\n text: `[error] ${message}`,\n });\n }\n\n // ───────── streaming ─────────\n\n async function streamSse(\n path: string,\n body: unknown,\n chat: ChatThread,\n turn: StoredTurn,\n ): Promise<void> {\n chat.abort = new AbortController();\n const resp = await fetch(`${apiBase}${path}`, {\n method: \"POST\",\n credentials: \"include\",\n headers: { \"Content-Type\": \"application/json\", Accept: \"text/event-stream\" },\n body: JSON.stringify(body),\n signal: chat.abort.signal,\n });\n if (!resp.ok || !resp.body) {\n throw new Error(`${resp.status} ${resp.statusText}`);\n }\n const reader = resp.body.getReader();\n const decoder = new TextDecoder();\n let buffer = \"\";\n /** Track whether the last applied event was a `text` event so we know\n * whether to merge subsequent text deltas into the same paragraph. */\n let lastWasText = false;\n\n while (true) {\n const { value, done } = await reader.read();\n if (done) break;\n buffer += decoder.decode(value, { stream: true });\n const events = buffer.split(\"\\n\\n\");\n buffer = events.pop() ?? \"\";\n for (const raw of events) {\n const dataLine = raw.split(\"\\n\").find((l) => l.startsWith(\"data: \"));\n if (!dataLine) continue;\n try {\n const evt = JSON.parse(dataLine.slice(6)) as StreamEvent;\n lastWasText = applyStreamEvent(evt, chat, turn, lastWasText);\n render();\n } catch (e) {\n console.warn(\"[coding-tab] bad sse event\", e);\n }\n }\n }\n }\n\n /**\n * Apply an SSE event into the active turn's timeline. Returns whether the\n * last applied event was a streaming text event (so the next text chunk can\n * be merged with it).\n */\n function applyStreamEvent(\n evt: StreamEvent,\n chat: ChatThread,\n turn: StoredTurn,\n lastWasText: boolean,\n ): boolean {\n switch (evt.kind) {\n case \"ready\":\n chat.activeRunId = evt.runId;\n return false;\n case \"text\": {\n // Each `text` SSE corresponds to one logical block from the model.\n // Within a single render burst the server sends one event per block,\n // so we always start a new paragraph here. (If the SDK ever streams\n // sub-block deltas, the server-side TurnBuffer will have already\n // coalesced them into one event for us.)\n const last = turn.events[turn.events.length - 1];\n if (lastWasText && last && last.kind === \"text\") {\n // Same delta train — append.\n last.text += evt.text;\n } else {\n turn.events.push({\n kind: \"text\",\n id: cryptoRandomId(),\n text: evt.text,\n });\n }\n return true;\n }\n case \"thinking\":\n return lastWasText;\n case \"tool\": {\n const existing = turn.events.find(\n (e): e is Extract<TimelineEvent, { kind: \"tool\" }> =>\n e.kind === \"tool\" && e.callId === evt.callId,\n );\n if (existing) {\n existing.status = evt.status;\n if (evt.args !== undefined) existing.args = evt.args;\n if (evt.result !== undefined) existing.result = evt.result;\n } else {\n turn.events.push({\n kind: \"tool\",\n id: cryptoRandomId(),\n callId: evt.callId,\n name: evt.name,\n status: evt.status,\n args: evt.args,\n result: evt.result,\n });\n }\n return false;\n }\n case \"status\":\n return lastWasText;\n case \"result\":\n turn.status = evt.status as TurnStatus;\n if (evt.pr) turn.pr = evt.pr;\n if (turn.isPlan && evt.status === \"finished\") turn.showExecute = true;\n chat.activeRunId = null;\n return false;\n case \"error\":\n turn.events.push({\n kind: \"text\",\n id: cryptoRandomId(),\n text: `[error] ${evt.message}`,\n });\n turn.status = \"error\";\n chat.activeRunId = null;\n return false;\n }\n }\n\n // ───────── bootstrap ─────────\n\n async function bootstrap() {\n try {\n const me = await api<MeResponse>(\"/auth/me\");\n state.me = me;\n if (me.defaultRepoUrl) {\n state.repoUrl = me.defaultRepoUrl;\n state.repoLocked = true;\n }\n } catch (err) {\n const status = (err as { status?: number }).status;\n if (status === 401 || status === 403) {\n state.me = null;\n render();\n return;\n }\n console.error(\"[coding-tab] /auth/me failed\", err);\n state.me = null;\n render();\n return;\n }\n try {\n const { models } = await api<{ models: ModelOption[] }>(\"/models\");\n state.models = models;\n if (models.length > 0 && !models.find((m) => m.choice === state.model)) {\n state.model = models[0]!.choice;\n }\n } catch (err) {\n console.error(\"[coding-tab] /models failed\", err);\n state.models = [\n { choice: \"sonnet\", cursorModelId: \"auto\", displayName: \"Sonnet (fallback)\" },\n { choice: \"opus\", cursorModelId: \"auto\", displayName: \"Opus (fallback)\" },\n ];\n }\n await loadChatList();\n if (state.chats.length > 0 && !state.activeChatId) {\n state.activeChatId = state.chats[0]!.meta.id;\n ensureChatLoaded(state.activeChatId).then(() => render());\n }\n render();\n }\n\n bootstrap();\n\n return {\n destroy() {\n for (const c of state.chats) c.abort?.abort();\n el.removeChild(root);\n },\n };\n}\n\nif (typeof window !== \"undefined\") {\n (window as unknown as { CodingTab?: { mountCodingTab: typeof mountCodingTab } }).CodingTab = {\n mountCodingTab,\n };\n}\n","/**\n * Tiny zero-dependency markdown renderer used for Plan-mode assistant turns.\n *\n * Supports headings (`#`/`##`/`###`/`####`), unordered (`-`/`*`) and ordered\n * (`1.`) lists, fenced code blocks, inline `code`, `**bold**`, `*italic*`,\n * `[link](url)`, horizontal rules (`---`), paragraphs, and hard breaks.\n *\n * All text is HTML-escaped before any markdown substitutions are applied so\n * the output is safe to inject via `innerHTML` even when the upstream model\n * dumps unsanitised user input back to us.\n */\n\nexport function escapeHtml(s: string): string {\n return s.replace(/[&<>\"']/g, (c) =>\n (\n {\n \"&\": \"&\",\n \"<\": \"<\",\n \">\": \">\",\n '\"': \""\",\n \"'\": \"'\",\n } as Record<string, string>\n )[c]!,\n );\n}\n\nfunction applyInline(text: string): string {\n // Inline code first — anything inside backticks should be inert.\n const segments: string[] = [];\n const codeRe = /`([^`\\n]+)`/g;\n let lastIndex = 0;\n let m: RegExpExecArray | null;\n while ((m = codeRe.exec(text)) !== null) {\n segments.push(applyInlineRest(text.slice(lastIndex, m.index)));\n segments.push(`<code>${m[1]!}</code>`);\n lastIndex = m.index + m[0].length;\n }\n segments.push(applyInlineRest(text.slice(lastIndex)));\n return segments.join(\"\");\n}\n\nfunction applyInlineRest(text: string): string {\n return text\n .replace(/\\*\\*([^*\\n]+)\\*\\*/g, \"<strong>$1</strong>\")\n .replace(/(^|[^*])\\*([^*\\n]+)\\*(?!\\*)/g, \"$1<em>$2</em>\")\n .replace(/\\[([^\\]\\n]+)\\]\\(([^)\\s]+)\\)/g, (_match, label: string, url: string) => {\n const safeUrl = /^(https?:|mailto:|\\/|#)/i.test(url) ? url : \"#\";\n return `<a href=\"${safeUrl}\" target=\"_blank\" rel=\"noreferrer noopener\">${label}</a>`;\n });\n}\n\ninterface ListContext {\n type: \"ul\" | \"ol\";\n indent: number;\n items: string[];\n}\n\nfunction flushParagraph(buf: string[], out: string[]) {\n if (buf.length === 0) return;\n const joined = buf.join(\" \").trim();\n buf.length = 0;\n if (!joined) return;\n out.push(`<p>${applyInline(joined)}</p>`);\n}\n\nfunction flushList(stack: ListContext[], out: string[]) {\n while (stack.length > 0) {\n const ctx = stack.pop()!;\n out.push(`<${ctx.type}>${ctx.items.join(\"\")}</${ctx.type}>`);\n }\n}\n\nexport function renderMarkdown(input: string): string {\n // Escape first so list markers / fences are still recognised but any HTML in\n // the source becomes inert.\n const escaped = escapeHtml(input).replace(/\\r\\n/g, \"\\n\");\n const lines = escaped.split(\"\\n\");\n\n const out: string[] = [];\n const paragraph: string[] = [];\n const listStack: ListContext[] = [];\n let inCode = false;\n let codeLang = \"\";\n const codeBuf: string[] = [];\n\n const closeOpenBlocks = () => {\n flushParagraph(paragraph, out);\n flushList(listStack, out);\n };\n\n for (let i = 0; i < lines.length; i++) {\n const line = lines[i]!;\n\n // Fenced code blocks\n const fence = line.match(/^\\s*```(\\w*)\\s*$/);\n if (fence) {\n if (inCode) {\n const cls = codeLang ? ` class=\"lang-${codeLang}\"` : \"\";\n out.push(`<pre><code${cls}>${codeBuf.join(\"\\n\")}</code></pre>`);\n codeBuf.length = 0;\n codeLang = \"\";\n inCode = false;\n } else {\n closeOpenBlocks();\n inCode = true;\n codeLang = fence[1] ?? \"\";\n }\n continue;\n }\n if (inCode) {\n codeBuf.push(line);\n continue;\n }\n\n // Blank line: paragraph break, list break.\n if (!line.trim()) {\n closeOpenBlocks();\n continue;\n }\n\n // Horizontal rule.\n if (/^\\s*(-{3,}|\\*{3,}|_{3,})\\s*$/.test(line)) {\n closeOpenBlocks();\n out.push(\"<hr />\");\n continue;\n }\n\n // Headings.\n const h = line.match(/^(#{1,6})\\s+(.*)$/);\n if (h) {\n closeOpenBlocks();\n const level = Math.min(6, h[1]!.length);\n out.push(`<h${level}>${applyInline(h[2]!.trim())}</h${level}>`);\n continue;\n }\n\n // List items.\n const ul = line.match(/^(\\s*)[-*]\\s+(.*)$/);\n const ol = line.match(/^(\\s*)(\\d+)\\.\\s+(.*)$/);\n if (ul || ol) {\n flushParagraph(paragraph, out);\n const indent = (ul ? ul[1] : ol![1])!.length;\n const type: \"ul\" | \"ol\" = ul ? \"ul\" : \"ol\";\n const content = (ul ? ul[2] : ol![3])!;\n // Pop deeper lists.\n while (listStack.length > 0 && listStack[listStack.length - 1]!.indent > indent) {\n const ctx = listStack.pop()!;\n const prev = listStack[listStack.length - 1]?.items;\n const html = `<${ctx.type}>${ctx.items.join(\"\")}</${ctx.type}>`;\n if (prev && prev.length > 0) {\n prev[prev.length - 1] = prev[prev.length - 1]!.replace(\n /<\\/li>$/,\n `${html}</li>`,\n );\n } else {\n out.push(html);\n }\n }\n const top = listStack[listStack.length - 1];\n if (!top || top.indent < indent || top.type !== type) {\n listStack.push({ type, indent, items: [`<li>${applyInline(content)}</li>`] });\n } else {\n top.items.push(`<li>${applyInline(content)}</li>`);\n }\n continue;\n }\n\n // If we were in a list and hit a non-list line, close the list.\n if (listStack.length > 0) {\n flushList(listStack, out);\n }\n paragraph.push(line.trim());\n }\n\n if (inCode) {\n out.push(`<pre><code>${codeBuf.join(\"\\n\")}</code></pre>`);\n } else {\n closeOpenBlocks();\n }\n\n return out.join(\"\\n\");\n}\n"],"mappings":"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","names":["tab_exports","__export","mountCodingTab","escapeHtml","s","c","applyInline","text","segments","codeRe","lastIndex","m","applyInlineRest","_match","label","url","flushParagraph","buf","out","joined","flushList","stack","ctx","renderMarkdown","input","lines","paragraph","listStack","inCode","codeLang","codeBuf","closeOpenBlocks","i","line","fence","cls","h","level","ul","ol","indent","type","content","prev","html","top","ICON_GITHUB","ICON_PLUS","ICON_TRASH","ICON_PENCIL","ICON_MENU","ICON_CHEVRON","escapeAttr","s","escapeHtml","repoSlug","url","m","cryptoRandomId","relativeTime","ts","diff","mountCodingTab","el","options","apiBase","root","stylesheetInjected","injectStylesheet","link","state","activeChat","c","api","path","init","resp","loadChatList","chats","existing","meta","prev","err","ensureChatLoaded","chatId","t","chat","createChat","thread","switchChat","render","deleteChat","renameChat","next","trimmed","headerHtml","sidebarHtml","renderChatRow","paneHtml","renderChatPane","renderEmptyPane","bindEvents","active","streaming","turns","threadHtml","renderTurn","placeholder","turn","text","extractText","eventsHtml","evt","renderEvent","statusBadge","planActions","prHtml","renderPr","isPlan","renderMarkdown","renderInlineText","renderToolEvent","para","p","summary","summarizeTool","expanded","detailHtml","renderToolDetail","name","args","a","lower","pattern","lines","argsBlock","formatBlob","resultBlock","value","pr","merge","mergeButton","statusLine","e","b","id","onSend","ev","onCancel","onExecute","onMerge","promptEl","prompt","userTurn","assistantTurn","streamSse","appendErrorToTurn","prUrl","result","message","body","reader","decoder","buffer","lastWasText","done","events","raw","dataLine","l","applyStreamEvent","last","bootstrap","me","status","models"]}
|
|
1
|
+
{"version":3,"sources":["../src/client/tab.ts","../src/client/markdown.ts"],"sourcesContent":["import { escapeHtml, renderMarkdown } from \"./markdown.js\";\nimport type {\n BranchInfo,\n ChatListItem,\n ChatMode,\n FullChat,\n MeResponse,\n ModelChoice,\n ModelOption,\n PrInfo,\n StoredChat,\n StoredTurn,\n StreamEvent,\n TimelineEvent,\n TurnStatus,\n} from \"../shared/types.js\";\n\ninterface MountOptions {\n apiBase: string;\n defaultRepo?: string;\n defaultRef?: string;\n defaultMode?: ChatMode;\n defaultModel?: ModelChoice;\n}\n\ninterface MountHandle {\n destroy(): void;\n}\n\n/**\n * Visible state of a \"Merge & Redeploy\" button. Stored per PR-url so the\n * feedback survives re-renders (which happen frequently during streaming) and\n * so multiple PRs in the same chat can have independent states.\n */\ntype MergeState =\n | { state: \"idle\" }\n | { state: \"loading\" }\n | { state: \"success\"; sha: string }\n | { state: \"error\"; message: string };\n\n/** Client-side ephemeral wrapper around a `StoredChat`. */\ninterface ChatThread {\n meta: ChatListItem;\n /** Loaded turns (after `/chats/:id` fetch). Undefined means not yet hydrated. */\n turns?: StoredTurn[];\n loaded: boolean;\n isStreaming: boolean;\n activeRunId: string | null;\n abort: AbortController | null;\n}\n\nconst ICON_GITHUB = `<svg width=\"18\" height=\"18\" viewBox=\"0 0 16 16\" fill=\"currentColor\"><path d=\"M8 0C3.58 0 0 3.58 0 8a8 8 0 005.47 7.59c.4.07.55-.17.55-.38v-1.34c-2.23.48-2.7-1.07-2.7-1.07-.36-.92-.89-1.16-.89-1.16-.73-.5.05-.49.05-.49.81.06 1.23.83 1.23.83.72 1.23 1.88.87 2.34.66.07-.52.28-.87.5-1.07-1.78-.2-3.65-.89-3.65-3.95 0-.87.31-1.59.82-2.15-.08-.2-.36-1.02.08-2.13 0 0 .67-.21 2.2.82a7.65 7.65 0 014 0c1.53-1.04 2.2-.82 2.2-.82.44 1.11.16 1.93.08 2.13.51.56.82 1.28.82 2.15 0 3.07-1.87 3.75-3.65 3.95.29.25.54.74.54 1.49v2.21c0 .21.15.46.55.38A8 8 0 0016 8c0-4.42-3.58-8-8-8z\"/></svg>`;\n\nconst ICON_PLUS = `<svg width=\"14\" height=\"14\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\"><path d=\"M8 3v10M3 8h10\"/></svg>`;\nconst ICON_TRASH = `<svg width=\"13\" height=\"13\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.4\" stroke-linecap=\"round\"><path d=\"M3 4h10M6 4V2.5a.5.5 0 01.5-.5h3a.5.5 0 01.5.5V4M5 4l.7 9.1a.9.9 0 00.9.9h2.8a.9.9 0 00.9-.9L11 4\"/></svg>`;\nconst ICON_PENCIL = `<svg width=\"13\" height=\"13\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.4\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M11 2.5l2.5 2.5L6 12.5 3 13l.5-3z\"/></svg>`;\nconst ICON_MENU = `<svg width=\"18\" height=\"18\" viewBox=\"0 0 16 16\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.6\" stroke-linecap=\"round\"><path d=\"M2.5 4h11M2.5 8h11M2.5 12h11\"/></svg>`;\nconst ICON_CHEVRON = `<svg class=\"ct-chevron\" width=\"10\" height=\"10\" viewBox=\"0 0 10 10\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1.6\" stroke-linecap=\"round\" stroke-linejoin=\"round\"><path d=\"M3 2l4 3-4 3\"/></svg>`;\n\nfunction escapeAttr(s: string): string {\n return escapeHtml(s);\n}\n\nfunction repoSlug(url: string): string {\n const m = url.match(/github\\.com[/:]([^/]+)\\/([^/]+?)(?:\\.git)?\\/?$/);\n return m ? `${m[1]}/${m[2]}` : url;\n}\n\nfunction repoWebUrl(url: string): string {\n // Strip a trailing `.git` and SSH-style `git@github.com:owner/repo.git` so\n // the URL is safe to drop into an `<a href>`.\n const m = url.match(/github\\.com[/:]([^/]+)\\/([^/]+?)(?:\\.git)?\\/?$/);\n return m ? `https://github.com/${m[1]}/${m[2]}` : url.replace(/\\.git\\/?$/, \"\");\n}\n\nfunction githubPullsUrl(url: string): string {\n return `${repoWebUrl(url)}/pulls`;\n}\n\nfunction cryptoRandomId(): string {\n return typeof crypto !== \"undefined\" && \"randomUUID\" in crypto\n ? crypto.randomUUID()\n : Math.random().toString(36).slice(2);\n}\n\nfunction relativeTime(ts: number): string {\n const diff = Date.now() - ts;\n if (diff < 60_000) return \"just now\";\n if (diff < 3_600_000) return `${Math.round(diff / 60_000)}m`;\n if (diff < 86_400_000) return `${Math.round(diff / 3_600_000)}h`;\n if (diff < 604_800_000) return `${Math.round(diff / 86_400_000)}d`;\n return new Date(ts).toLocaleDateString();\n}\n\nexport function mountCodingTab(el: HTMLElement, options: MountOptions): MountHandle {\n const apiBase = options.apiBase.replace(/\\/$/, \"\");\n const root = document.createElement(\"div\");\n root.className = \"coding-tab\";\n el.innerHTML = \"\";\n el.appendChild(root);\n\n let stylesheetInjected = false;\n function injectStylesheet() {\n if (stylesheetInjected) return;\n stylesheetInjected = true;\n if (document.querySelector(`link[data-coding-tab=\"style\"]`)) return;\n const link = document.createElement(\"link\");\n link.rel = \"stylesheet\";\n link.href = `${apiBase}/style.css`;\n link.dataset.codingTab = \"style\";\n document.head.appendChild(link);\n }\n injectStylesheet();\n\n const state: {\n me: MeResponse | null;\n models: ModelOption[];\n mode: ChatMode;\n model: ModelChoice;\n repoUrl: string;\n repoLocked: boolean;\n chats: ChatThread[];\n activeChatId: string | null;\n sidebarOpen: boolean;\n /** Tracks expanded tool events across re-renders (keyed by `event.id`). */\n expandedTools: Set<string>;\n /** Per-PR merge state so the button shows loading/success/error feedback. */\n mergeStates: Map<string, MergeState>;\n } = {\n me: null,\n models: [],\n mode: options.defaultMode ?? \"plan\",\n model: options.defaultModel ?? \"sonnet\",\n repoUrl: options.defaultRepo ?? \"\",\n repoLocked: false,\n chats: [],\n activeChatId: null,\n sidebarOpen: window.innerWidth >= 720,\n expandedTools: new Set(),\n mergeStates: new Map(),\n };\n\n function activeChat(): ChatThread | null {\n return state.chats.find((c) => c.meta.id === state.activeChatId) ?? null;\n }\n\n async function api<T = unknown>(path: string, init?: RequestInit): Promise<T> {\n const resp = await fetch(`${apiBase}${path}`, {\n credentials: \"include\",\n headers: { \"Content-Type\": \"application/json\", ...(init?.headers ?? {}) },\n ...init,\n });\n if (!resp.ok)\n throw Object.assign(new Error(`${resp.status} ${resp.statusText}`), { status: resp.status });\n return (await resp.json()) as T;\n }\n\n // ───────── chat lifecycle ─────────\n\n async function loadChatList(): Promise<void> {\n try {\n const { chats } = await api<{ chats: ChatListItem[] }>(\"/chats\");\n // Preserve runtime state for chats we already had.\n const existing = new Map(state.chats.map((c) => [c.meta.id, c]));\n state.chats = chats.map((meta) => {\n const prev = existing.get(meta.id);\n return prev\n ? { ...prev, meta }\n : {\n meta,\n loaded: false,\n isStreaming: false,\n activeRunId: null,\n abort: null,\n };\n });\n } catch (err) {\n console.error(\"[coding-tab] /chats list failed\", err);\n state.chats = [];\n }\n }\n\n async function ensureChatLoaded(chatId: string): Promise<ChatThread | null> {\n const t = state.chats.find((c) => c.meta.id === chatId);\n if (!t) return null;\n if (t.loaded) return t;\n try {\n const { chat } = await api<{ chat: FullChat }>(`/chats/${chatId}`);\n t.turns = chat.turns;\n t.loaded = true;\n t.meta = {\n id: chat.id,\n title: chat.title,\n mode: chat.mode,\n model: chat.model,\n createdAt: chat.createdAt,\n updatedAt: chat.updatedAt,\n };\n // Sync composer state to whatever this chat last used.\n state.mode = chat.mode;\n state.model = chat.model;\n return t;\n } catch (err) {\n console.error(`[coding-tab] /chats/${chatId} load failed`, err);\n return null;\n }\n }\n\n async function createChat(): Promise<ChatThread | null> {\n try {\n const { chat } = await api<{ chat: StoredChat }>(\"/chats\", {\n method: \"POST\",\n body: JSON.stringify({\n mode: state.mode,\n model: state.model,\n repoUrl: state.repoLocked ? undefined : state.repoUrl || undefined,\n }),\n });\n const thread: ChatThread = {\n meta: {\n id: chat.id,\n title: chat.title,\n mode: chat.mode,\n model: chat.model,\n createdAt: chat.createdAt,\n updatedAt: chat.updatedAt,\n },\n turns: [],\n loaded: true,\n isStreaming: false,\n activeRunId: null,\n abort: null,\n };\n state.chats.unshift(thread);\n state.activeChatId = thread.meta.id;\n return thread;\n } catch (err) {\n console.error(\"[coding-tab] create chat failed\", err);\n return null;\n }\n }\n\n async function switchChat(chatId: string): Promise<void> {\n state.activeChatId = chatId;\n if (window.innerWidth < 720) state.sidebarOpen = false;\n render();\n await ensureChatLoaded(chatId);\n render();\n }\n\n async function deleteChat(chatId: string): Promise<void> {\n const thread = state.chats.find((c) => c.meta.id === chatId);\n if (!thread) return;\n if (!confirm(`Delete chat \"${thread.meta.title}\"? This cannot be undone.`)) return;\n thread.abort?.abort();\n try {\n await api(`/chats/${chatId}`, { method: \"DELETE\" });\n } catch (err) {\n console.error(`[coding-tab] delete chat ${chatId} failed`, err);\n return;\n }\n state.chats = state.chats.filter((c) => c.meta.id !== chatId);\n if (state.activeChatId === chatId) {\n state.activeChatId = state.chats[0]?.meta.id ?? null;\n }\n render();\n }\n\n async function renameChat(chatId: string): Promise<void> {\n const thread = state.chats.find((c) => c.meta.id === chatId);\n if (!thread) return;\n const next = prompt(\"Rename chat\", thread.meta.title);\n if (next === null) return;\n const trimmed = next.trim();\n if (!trimmed || trimmed === thread.meta.title) return;\n try {\n const { chat } = await api<{ chat: StoredChat }>(`/chats/${chatId}`, {\n method: \"PATCH\",\n body: JSON.stringify({ title: trimmed }),\n });\n thread.meta = { ...thread.meta, title: chat.title, updatedAt: chat.updatedAt };\n render();\n } catch (err) {\n console.error(`[coding-tab] rename chat ${chatId} failed`, err);\n }\n }\n\n // ───────── render ─────────\n\n function render() {\n if (!state.me) {\n root.innerHTML = `\n <div class=\"coding-tab__signin\">\n <h2>Coding Tab</h2>\n <p>Sign in with GitHub to start a coding session against your repo. Your token is used to clone, push, and open pull requests on your behalf.</p>\n <a href=\"${apiBase}/auth/login\">${ICON_GITHUB}<span>Sign in with GitHub</span></a>\n </div>\n `;\n return;\n }\n\n const headerHtml = `\n <div class=\"coding-tab__header\">\n <button class=\"coding-tab__hamburger\" data-role=\"toggle-sidebar\" aria-label=\"Toggle chat list\">${ICON_MENU}</button>\n <select data-role=\"model\" title=\"Model\">\n ${state.models\n .map(\n (m) =>\n `<option value=\"${m.choice}\" ${state.model === m.choice ? \"selected\" : \"\"}>${escapeHtml(m.displayName)}</option>`,\n )\n .join(\"\")}\n </select>\n <div class=\"coding-tab__mode\">\n <button data-mode=\"plan\" class=\"${state.mode === \"plan\" ? \"is-active\" : \"\"}\">Plan</button>\n <button data-mode=\"agent\" class=\"${state.mode === \"agent\" ? \"is-active\" : \"\"}\">Agent</button>\n </div>\n ${\n state.repoLocked && state.repoUrl\n ? `<a class=\"coding-tab__repo-locked\" href=\"${escapeAttr(githubPullsUrl(state.repoUrl))}\" target=\"_blank\" rel=\"noreferrer\" title=\"Open ${escapeAttr(repoSlug(state.repoUrl))} pull requests on GitHub\">${ICON_GITHUB}<span>${escapeHtml(repoSlug(state.repoUrl))}</span></a>`\n : `<input data-role=\"repo\" placeholder=\"https://github.com/org/repo\" value=\"${escapeAttr(state.repoUrl)}\" style=\"flex:1;min-width:200px\" />`\n }\n <div class=\"coding-tab__user\">\n ${state.me.avatarUrl ? `<img src=\"${state.me.avatarUrl}\" alt=\"\" />` : \"\"}\n <span class=\"coding-tab__user-name\">@${escapeHtml(state.me.githubLogin)}</span>\n <button data-role=\"logout\">Sign out</button>\n </div>\n </div>\n `;\n\n const sidebarHtml = `\n <aside class=\"coding-tab__sidebar ${state.sidebarOpen ? \"is-open\" : \"\"}\">\n <button class=\"coding-tab__btn primary coding-tab__new-chat\" data-role=\"new-chat\">${ICON_PLUS}<span>New chat</span></button>\n <div class=\"coding-tab__chat-list\">\n ${\n state.chats.length === 0\n ? `<div class=\"coding-tab__chat-empty\">No chats yet — start one with the button above.</div>`\n : state.chats.map((c) => renderChatRow(c)).join(\"\")\n }\n </div>\n </aside>\n `;\n\n const chat = activeChat();\n const paneHtml = `\n <main class=\"coding-tab__pane\">\n ${chat ? renderChatPane(chat) : renderEmptyPane()}\n </main>\n `;\n\n root.innerHTML = `\n ${headerHtml}\n <div class=\"coding-tab__body\">\n ${sidebarHtml}\n ${paneHtml}\n </div>\n `;\n\n const thread = root.querySelector(\"[data-role=thread]\") as HTMLElement | null;\n if (thread) thread.scrollTop = thread.scrollHeight;\n\n bindEvents();\n }\n\n function renderChatRow(c: ChatThread): string {\n const active = c.meta.id === state.activeChatId ? \" is-active\" : \"\";\n const streaming = c.isStreaming ? `<span class=\"coding-tab__chat-row-dot\" title=\"Streaming\"></span>` : \"\";\n return `\n <div class=\"coding-tab__chat-row${active}\" data-chat-id=\"${c.meta.id}\">\n <button class=\"coding-tab__chat-row-main\" data-role=\"select-chat\" data-chat-id=\"${c.meta.id}\">\n ${streaming}\n <span class=\"coding-tab__chat-row-title\">${escapeHtml(c.meta.title)}</span>\n <span class=\"coding-tab__chat-row-meta\">${c.meta.mode === \"plan\" ? \"Plan\" : \"Agent\"} · ${relativeTime(c.meta.updatedAt)}</span>\n </button>\n <div class=\"coding-tab__chat-row-actions\">\n <button data-role=\"rename-chat\" data-chat-id=\"${c.meta.id}\" title=\"Rename\">${ICON_PENCIL}</button>\n <button data-role=\"delete-chat\" data-chat-id=\"${c.meta.id}\" title=\"Delete\">${ICON_TRASH}</button>\n </div>\n </div>\n `;\n }\n\n function renderEmptyPane(): string {\n return `\n <div class=\"coding-tab__pane-empty\">\n <h3>Start a new chat</h3>\n <p>Pick a mode (Plan or Agent), choose a model, and click <strong>New chat</strong> to begin. Each chat keeps its own context.</p>\n <button class=\"coding-tab__btn primary\" data-role=\"new-chat\">${ICON_PLUS}<span>New chat</span></button>\n </div>\n `;\n }\n\n function renderChatPane(chat: ChatThread): string {\n const turns = chat.turns ?? [];\n const threadHtml =\n !chat.loaded\n ? `<div class=\"coding-tab__notice info\">Loading…</div>`\n : turns.length === 0\n ? `<div class=\"coding-tab__notice info\">Type a message below to kick off this chat.</div>`\n : turns.map((t) => renderTurn(t)).join(\"\");\n\n const placeholder =\n turns.length === 0\n ? `Ask anything (e.g. \"add dark mode that follows the OS setting\")`\n : state.mode === \"plan\"\n ? \"Refine the plan, or ask another planning question\"\n : \"Send another instruction\";\n\n return `\n <div class=\"coding-tab__thread\" data-role=\"thread\">${threadHtml}</div>\n <div class=\"coding-tab__composer\">\n <textarea data-role=\"prompt\" placeholder=\"${escapeAttr(placeholder)}\" rows=\"2\"></textarea>\n <button class=\"coding-tab__btn primary\" data-role=\"send\" ${chat.isStreaming ? \"disabled\" : \"\"}>${chat.isStreaming ? \"Working…\" : \"Send\"}</button>\n ${chat.isStreaming ? `<button class=\"coding-tab__btn danger\" data-role=\"cancel\">Stop</button>` : \"\"}\n </div>\n `;\n }\n\n function renderTurn(turn: StoredTurn): string {\n if (turn.role === \"user\") {\n const text = turn.prompt ?? extractText(turn);\n return `<div class=\"coding-tab__msg user\">\n <div class=\"coding-tab__msg-role\">You · ${turn.isPlan ? \"Plan\" : \"Agent\"}</div>\n <div class=\"coding-tab__msg-body\">${escapeHtml(text)}</div>\n </div>`;\n }\n\n const eventsHtml =\n turn.events.length === 0 && turn.status === \"running\"\n ? `<div class=\"coding-tab__msg-pending\">Working…</div>`\n : turn.events.map((evt) => renderEvent(evt, turn.isPlan)).join(\"\");\n\n const statusBadge = turn.status && turn.status !== \"finished\"\n ? `<div class=\"coding-tab__status\">${escapeHtml(turn.status)}</div>`\n : \"\";\n\n const planActions =\n turn.showExecute && !activeChat()?.isStreaming\n ? `<div class=\"coding-tab__plan-actions\">\n <button class=\"coding-tab__btn primary\" data-role=\"execute\" data-turn=\"${turn.id}\">Execute plan</button>\n </div>`\n : \"\";\n\n const prHtml = turn.pr\n ? renderPr(turn.pr)\n : turn.branch\n ? renderBranchFallback(turn.branch)\n : \"\";\n\n return `<div class=\"coding-tab__msg assistant\">\n <div class=\"coding-tab__msg-role\">Coding Tab · ${turn.isPlan ? \"Plan\" : \"Agent\"} ${statusBadge}</div>\n <div class=\"coding-tab__msg-body\">${eventsHtml}${planActions}${prHtml}</div>\n </div>`;\n }\n\n function renderEvent(evt: TimelineEvent, isPlan: boolean): string {\n if (evt.kind === \"text\") {\n const text = evt.text.trim();\n if (!text) return \"\";\n const html = isPlan ? renderMarkdown(text) : renderInlineText(text);\n return `<div class=\"coding-tab__md\">${html}</div>`;\n }\n return renderToolEvent(evt);\n }\n\n function renderInlineText(text: string): string {\n // Lightweight rendering for Agent-mode messages: keep paragraph breaks,\n // inline `code`, **bold**, [text](url) markdown links, and bare-URL\n // autolinks. We deliberately skip headings/lists so the model's casual\n // streaming output doesn't get over-formatted.\n return escapeHtml(text)\n .split(/\\n\\s*\\n/)\n .map((para) =>\n para\n .replace(/`([^`\\n]+)`/g, \"<code>$1</code>\")\n .replace(/\\*\\*([^*\\n]+)\\*\\*/g, \"<strong>$1</strong>\")\n // Markdown links first so we don't double-link the URL inside.\n .replace(/\\[([^\\]\\n]+)\\]\\(([^)\\s]+)\\)/g, (_m, label: string, url: string) => {\n const safe = /^(https?:|mailto:|\\/|#)/i.test(url) ? url : \"#\";\n return `<a href=\"${safe}\" target=\"_blank\" rel=\"noreferrer noopener\">${label}</a>`;\n })\n // Then autolink bare URLs that aren't already inside an href=\"...\".\n .replace(/(^|[\\s(])(https?:\\/\\/[^\\s<>\"')\\]]+)/g, (_m, lead: string, url: string) => {\n return `${lead}<a href=\"${url}\" target=\"_blank\" rel=\"noreferrer noopener\">${url}</a>`;\n })\n .replace(/\\n/g, \"<br />\"),\n )\n .map((p) => `<p>${p}</p>`)\n .join(\"\");\n }\n\n function renderToolEvent(\n evt: Extract<TimelineEvent, { kind: \"tool\" }>,\n ): string {\n const summary = summarizeTool(evt.name, evt.args);\n const expanded = state.expandedTools.has(evt.id);\n const detailHtml = expanded ? renderToolDetail(evt) : \"\";\n return `\n <div class=\"coding-tab__tool\" data-status=\"${evt.status}\" data-tool-id=\"${evt.id}\">\n <button class=\"coding-tab__tool-header\" data-role=\"toggle-tool\" data-tool-id=\"${evt.id}\" aria-expanded=\"${expanded}\">\n <span class=\"coding-tab__tool-dot\"></span>\n ${ICON_CHEVRON}\n <span class=\"coding-tab__tool-name\">${escapeHtml(evt.name)}</span>\n ${summary ? `<span class=\"coding-tab__tool-summary\">${escapeHtml(summary)}</span>` : \"\"}\n </button>\n ${detailHtml}\n </div>\n `;\n }\n\n function summarizeTool(name: string, args: unknown): string {\n if (!args || typeof args !== \"object\") return \"\";\n const a = args as Record<string, unknown>;\n const lower = name.toLowerCase();\n if (lower.includes(\"grep\")) {\n const pattern = typeof a.pattern === \"string\" ? a.pattern : typeof a.query === \"string\" ? a.query : \"\";\n return pattern ? `\"${pattern}\"` : \"\";\n }\n if (lower.includes(\"read\") && typeof a.path === \"string\") {\n const lines =\n typeof a.offset === \"number\" || typeof a.limit === \"number\"\n ? `:${a.offset ?? \"\"}-${(Number(a.offset ?? 0) + Number(a.limit ?? 0)) || \"\"}`\n : \"\";\n return `${a.path}${lines}`;\n }\n if ((lower.includes(\"glob\") || lower.includes(\"file_search\")) && typeof a.glob_pattern === \"string\") {\n return a.glob_pattern as string;\n }\n if (lower === \"shell\" && typeof a.command === \"string\") {\n return (a.command as string).slice(0, 80);\n }\n if (lower === \"task\" && typeof a.description === \"string\") {\n return a.description as string;\n }\n if (lower.includes(\"write\") && typeof a.path === \"string\") {\n return a.path as string;\n }\n if (lower.includes(\"edit\") && typeof a.path === \"string\") {\n return a.path as string;\n }\n if (lower.includes(\"delete\") && typeof a.path === \"string\") {\n return a.path as string;\n }\n return \"\";\n }\n\n function renderToolDetail(evt: Extract<TimelineEvent, { kind: \"tool\" }>): string {\n const argsBlock =\n evt.args !== undefined && evt.args !== null\n ? `<div class=\"coding-tab__tool-section\"><div class=\"coding-tab__tool-label\">Args</div><pre>${escapeHtml(formatBlob(evt.args))}</pre></div>`\n : \"\";\n const resultBlock =\n evt.result !== undefined && evt.result !== null\n ? `<div class=\"coding-tab__tool-section\"><div class=\"coding-tab__tool-label\">Result</div><pre>${escapeHtml(formatBlob(evt.result))}</pre></div>`\n : evt.status === \"running\"\n ? `<div class=\"coding-tab__tool-section\"><div class=\"coding-tab__tool-label\">Result</div><pre class=\"coding-tab__tool-pending\">…running</pre></div>`\n : \"\";\n return `<div class=\"coding-tab__tool-detail\">${argsBlock}${resultBlock}</div>`;\n }\n\n function formatBlob(value: unknown): string {\n if (typeof value === \"string\") return value;\n try {\n return JSON.stringify(value, null, 2);\n } catch {\n return String(value);\n }\n }\n\n function renderBranchFallback(branch: BranchInfo): string {\n // Surfaced when the agent pushed commits but Cursor's auto-PR step didn't\n // create a PR (typically because of an account/team-level setting). Lets\n // the user open the PR themselves with one click instead of being stuck.\n const slug = repoSlug(branch.repoUrl);\n return `<div class=\"coding-tab__pr coding-tab__pr--branch\">\n <div class=\"coding-tab__pr-title\">Branch pushed: <code>${escapeHtml(branch.branch)}</code></div>\n <div class=\"coding-tab__pr-status\">No PR was opened automatically. Click below to create one yourself on GitHub.</div>\n <div class=\"coding-tab__pr-actions\">\n <a class=\"coding-tab__btn primary\" href=\"${escapeAttr(branch.compareUrl)}\" target=\"_blank\" rel=\"noreferrer\">Create PR on GitHub</a>\n <a class=\"coding-tab__btn\" href=\"${escapeAttr(branch.repoUrl)}/pulls\" target=\"_blank\" rel=\"noreferrer\">View ${escapeHtml(slug)} PRs</a>\n </div>\n </div>`;\n }\n\n function renderPr(pr: PrInfo): string {\n const merge = state.mergeStates.get(pr.url) ?? { state: \"idle\" as const };\n let mergeButton: string;\n let statusLine = \"\";\n switch (merge.state) {\n case \"loading\":\n mergeButton = `<button class=\"coding-tab__btn primary\" data-state=\"loading\" disabled>Merging…</button>`;\n statusLine = `<div class=\"coding-tab__pr-status\">Merging this PR and triggering Railway redeploy…</div>`;\n break;\n case \"success\":\n mergeButton = `<button class=\"coding-tab__btn success\" disabled>Merged ✓</button>`;\n statusLine = `<div class=\"coding-tab__pr-status is-success\">Merged commit ${escapeHtml(merge.sha.slice(0, 7))}. Railway should redeploy on the next push hook.</div>`;\n break;\n case \"error\":\n mergeButton = `<button class=\"coding-tab__btn primary\" data-role=\"merge\" data-pr=\"${escapeAttr(pr.url)}\">Try merge again</button>`;\n statusLine = `<div class=\"coding-tab__pr-status is-error\">Merge failed: ${escapeHtml(merge.message)}</div>`;\n break;\n default:\n mergeButton = `<button class=\"coding-tab__btn primary\" data-role=\"merge\" data-pr=\"${escapeAttr(pr.url)}\">Merge & Redeploy</button>`;\n }\n return `<div class=\"coding-tab__pr\">\n <div class=\"coding-tab__pr-title\">PR #${pr.number} opened in ${escapeHtml(pr.owner)}/${escapeHtml(pr.repo)}</div>\n ${pr.title ? `<div>${escapeHtml(pr.title)}</div>` : \"\"}\n <div class=\"coding-tab__pr-actions\">\n <a class=\"coding-tab__btn\" href=\"${escapeAttr(pr.url)}\" target=\"_blank\" rel=\"noreferrer\">Open in GitHub</a>\n ${mergeButton}\n </div>\n ${statusLine}\n </div>`;\n }\n\n function extractText(turn: StoredTurn): string {\n return turn.events\n .filter((e): e is Extract<TimelineEvent, { kind: \"text\" }> => e.kind === \"text\")\n .map((e) => e.text)\n .join(\"\\n\\n\");\n }\n\n // ───────── events ─────────\n\n function bindEvents() {\n root.querySelector(`[data-role=\"model\"]`)?.addEventListener(\"change\", (e) => {\n state.model = (e.target as HTMLSelectElement).value as ModelChoice;\n });\n root.querySelectorAll(`[data-mode]`).forEach((b) => {\n b.addEventListener(\"click\", () => {\n state.mode = (b as HTMLElement).dataset.mode as ChatMode;\n render();\n });\n });\n root.querySelector(`[data-role=\"repo\"]`)?.addEventListener(\"change\", (e) => {\n state.repoUrl = (e.target as HTMLInputElement).value.trim();\n });\n root.querySelector(`[data-role=\"logout\"]`)?.addEventListener(\"click\", async () => {\n await fetch(`${apiBase}/auth/logout`, { method: \"POST\", credentials: \"include\" }).catch(() => {});\n for (const c of state.chats) c.abort?.abort();\n state.me = null;\n state.chats = [];\n state.activeChatId = null;\n render();\n });\n root.querySelector(`[data-role=\"toggle-sidebar\"]`)?.addEventListener(\"click\", () => {\n state.sidebarOpen = !state.sidebarOpen;\n render();\n });\n\n root.querySelectorAll(`[data-role=\"new-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", async () => {\n const t = await createChat();\n if (t && window.innerWidth < 720) state.sidebarOpen = false;\n render();\n }),\n );\n root.querySelectorAll(`[data-role=\"select-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", () => {\n const id = (b as HTMLElement).dataset.chatId!;\n switchChat(id);\n }),\n );\n root.querySelectorAll(`[data-role=\"rename-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", (e) => {\n e.stopPropagation();\n renameChat((b as HTMLElement).dataset.chatId!);\n }),\n );\n root.querySelectorAll(`[data-role=\"delete-chat\"]`).forEach((b) =>\n b.addEventListener(\"click\", (e) => {\n e.stopPropagation();\n deleteChat((b as HTMLElement).dataset.chatId!);\n }),\n );\n\n root.querySelector(`[data-role=\"send\"]`)?.addEventListener(\"click\", () => onSend());\n root.querySelector(`[data-role=\"prompt\"]`)?.addEventListener(\"keydown\", (e: Event) => {\n const ev = e as KeyboardEvent;\n if (ev.key === \"Enter\" && (ev.metaKey || ev.ctrlKey)) {\n ev.preventDefault();\n onSend();\n }\n });\n root.querySelector(`[data-role=\"cancel\"]`)?.addEventListener(\"click\", () => onCancel());\n root.querySelectorAll(`[data-role=\"execute\"]`).forEach((b) =>\n b.addEventListener(\"click\", () => onExecute()),\n );\n root.querySelectorAll(`[data-role=\"merge\"]`).forEach((b) => {\n b.addEventListener(\"click\", () => onMerge((b as HTMLElement).dataset.pr!));\n });\n\n root.querySelectorAll(`[data-role=\"toggle-tool\"]`).forEach((b) =>\n b.addEventListener(\"click\", () => {\n const id = (b as HTMLElement).dataset.toolId!;\n if (state.expandedTools.has(id)) state.expandedTools.delete(id);\n else state.expandedTools.add(id);\n render();\n }),\n );\n }\n\n // ───────── send / execute / cancel / merge ─────────\n\n async function onSend() {\n let chat = activeChat();\n if (!chat) {\n chat = await createChat();\n if (!chat) return;\n }\n if (chat.isStreaming) return;\n\n const promptEl = root.querySelector(`[data-role=\"prompt\"]`) as HTMLTextAreaElement | null;\n const prompt = promptEl?.value.trim() ?? \"\";\n if (!prompt) return;\n if (promptEl) promptEl.value = \"\";\n\n chat.turns = chat.turns ?? [];\n const userTurn: StoredTurn = {\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"user\",\n isPlan: state.mode === \"plan\",\n status: \"finished\",\n events: [{ kind: \"text\", id: cryptoRandomId(), text: prompt }],\n prompt,\n createdAt: Date.now(),\n };\n chat.turns.push(userTurn);\n\n const assistantTurn: StoredTurn = {\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"assistant\",\n isPlan: state.mode === \"plan\",\n status: \"running\",\n events: [],\n createdAt: Date.now(),\n };\n chat.turns.push(assistantTurn);\n chat.isStreaming = true;\n if (chat.meta.title === \"New chat\") {\n chat.meta = { ...chat.meta, title: prompt.length > 60 ? `${prompt.slice(0, 57)}…` : prompt };\n }\n chat.meta = { ...chat.meta, mode: state.mode, model: state.model, updatedAt: Date.now() };\n render();\n\n try {\n await streamSse(\n \"/agent/send\",\n {\n chatId: chat.meta.id,\n prompt,\n mode: state.mode,\n },\n chat,\n assistantTurn,\n );\n } catch (err) {\n if ((err as Error).name !== \"AbortError\") {\n appendErrorToTurn(assistantTurn, err);\n }\n assistantTurn.status = \"error\";\n } finally {\n chat.isStreaming = false;\n chat.activeRunId = null;\n chat.abort = null;\n render();\n }\n }\n\n async function onExecute() {\n const chat = activeChat();\n if (!chat || chat.isStreaming) return;\n chat.turns = chat.turns ?? [];\n const turn: StoredTurn = {\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"assistant\",\n isPlan: false,\n status: \"running\",\n events: [],\n createdAt: Date.now(),\n };\n chat.turns.push(turn);\n chat.isStreaming = true;\n render();\n try {\n await streamSse(\"/agent/execute\", { chatId: chat.meta.id }, chat, turn);\n } catch (err) {\n if ((err as Error).name !== \"AbortError\") {\n appendErrorToTurn(turn, err);\n }\n turn.status = \"error\";\n } finally {\n chat.isStreaming = false;\n chat.activeRunId = null;\n chat.abort = null;\n render();\n }\n }\n\n async function onCancel() {\n const chat = activeChat();\n if (!chat) return;\n if (chat.activeRunId) {\n try {\n await api(\"/agent/cancel\", {\n method: \"POST\",\n body: JSON.stringify({ chatId: chat.meta.id, runId: chat.activeRunId }),\n });\n } catch (err) {\n console.error(\"[coding-tab] cancel failed\", err);\n }\n }\n chat.abort?.abort();\n }\n\n async function onMerge(prUrl: string) {\n const chat = activeChat();\n if (!chat) return;\n if (state.mergeStates.get(prUrl)?.state === \"loading\") return;\n state.mergeStates.set(prUrl, { state: \"loading\" });\n render();\n\n chat.turns = chat.turns ?? [];\n try {\n const result = await api<{ sha: string; merged: boolean }>(\"/pr/merge\", {\n method: \"POST\",\n body: JSON.stringify({ prUrl, mergeMethod: \"squash\" }),\n });\n if (result.merged) {\n state.mergeStates.set(prUrl, { state: \"success\", sha: result.sha });\n chat.turns.push({\n id: cryptoRandomId(),\n chatId: chat.meta.id,\n seq: chat.turns.length,\n role: \"assistant\",\n isPlan: false,\n status: \"finished\",\n events: [\n {\n kind: \"text\",\n id: cryptoRandomId(),\n text: `Merged commit \\`${result.sha.slice(0, 7)}\\` into the default branch. Railway should redeploy on the next push hook.`,\n },\n ],\n createdAt: Date.now(),\n });\n } else {\n state.mergeStates.set(prUrl, {\n state: \"error\",\n message: \"GitHub returned merged=false\",\n });\n }\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err);\n state.mergeStates.set(prUrl, { state: \"error\", message });\n } finally {\n render();\n }\n }\n\n function appendErrorToTurn(turn: StoredTurn, err: unknown): void {\n const message = err instanceof Error ? err.message : String(err);\n turn.events.push({\n kind: \"text\",\n id: cryptoRandomId(),\n text: `[error] ${message}`,\n });\n }\n\n // ───────── streaming ─────────\n\n async function streamSse(\n path: string,\n body: unknown,\n chat: ChatThread,\n turn: StoredTurn,\n ): Promise<void> {\n chat.abort = new AbortController();\n const resp = await fetch(`${apiBase}${path}`, {\n method: \"POST\",\n credentials: \"include\",\n headers: { \"Content-Type\": \"application/json\", Accept: \"text/event-stream\" },\n body: JSON.stringify(body),\n signal: chat.abort.signal,\n });\n if (!resp.ok || !resp.body) {\n throw new Error(`${resp.status} ${resp.statusText}`);\n }\n const reader = resp.body.getReader();\n const decoder = new TextDecoder();\n let buffer = \"\";\n /** Track whether the last applied event was a `text` event so we know\n * whether to merge subsequent text deltas into the same paragraph. */\n let lastWasText = false;\n\n while (true) {\n const { value, done } = await reader.read();\n if (done) break;\n buffer += decoder.decode(value, { stream: true });\n const events = buffer.split(\"\\n\\n\");\n buffer = events.pop() ?? \"\";\n for (const raw of events) {\n const dataLine = raw.split(\"\\n\").find((l) => l.startsWith(\"data: \"));\n if (!dataLine) continue;\n try {\n const evt = JSON.parse(dataLine.slice(6)) as StreamEvent;\n lastWasText = applyStreamEvent(evt, chat, turn, lastWasText);\n render();\n } catch (e) {\n console.warn(\"[coding-tab] bad sse event\", e);\n }\n }\n }\n }\n\n /**\n * Apply an SSE event into the active turn's timeline. Returns whether the\n * last applied event was a streaming text event (so the next text chunk can\n * be merged with it).\n */\n function applyStreamEvent(\n evt: StreamEvent,\n chat: ChatThread,\n turn: StoredTurn,\n lastWasText: boolean,\n ): boolean {\n switch (evt.kind) {\n case \"ready\":\n chat.activeRunId = evt.runId;\n return false;\n case \"text\": {\n // Each `text` SSE corresponds to one logical block from the model.\n // Within a single render burst the server sends one event per block,\n // so we always start a new paragraph here. (If the SDK ever streams\n // sub-block deltas, the server-side TurnBuffer will have already\n // coalesced them into one event for us.)\n const last = turn.events[turn.events.length - 1];\n if (lastWasText && last && last.kind === \"text\") {\n // Same delta train — append.\n last.text += evt.text;\n } else {\n turn.events.push({\n kind: \"text\",\n id: cryptoRandomId(),\n text: evt.text,\n });\n }\n return true;\n }\n case \"thinking\":\n return lastWasText;\n case \"tool\": {\n const existing = turn.events.find(\n (e): e is Extract<TimelineEvent, { kind: \"tool\" }> =>\n e.kind === \"tool\" && e.callId === evt.callId,\n );\n if (existing) {\n existing.status = evt.status;\n if (evt.args !== undefined) existing.args = evt.args;\n if (evt.result !== undefined) existing.result = evt.result;\n } else {\n turn.events.push({\n kind: \"tool\",\n id: cryptoRandomId(),\n callId: evt.callId,\n name: evt.name,\n status: evt.status,\n args: evt.args,\n result: evt.result,\n });\n }\n return false;\n }\n case \"status\":\n return lastWasText;\n case \"result\":\n turn.status = evt.status as TurnStatus;\n if (evt.pr) turn.pr = evt.pr;\n if (evt.branch && !evt.pr) turn.branch = evt.branch;\n if (turn.isPlan && evt.status === \"finished\") turn.showExecute = true;\n chat.activeRunId = null;\n return false;\n case \"error\":\n turn.events.push({\n kind: \"text\",\n id: cryptoRandomId(),\n text: `[error] ${evt.message}`,\n });\n turn.status = \"error\";\n chat.activeRunId = null;\n return false;\n }\n }\n\n // ───────── bootstrap ─────────\n\n async function bootstrap() {\n try {\n const me = await api<MeResponse>(\"/auth/me\");\n state.me = me;\n if (me.defaultRepoUrl) {\n state.repoUrl = me.defaultRepoUrl;\n state.repoLocked = true;\n }\n } catch (err) {\n const status = (err as { status?: number }).status;\n if (status === 401 || status === 403) {\n state.me = null;\n render();\n return;\n }\n console.error(\"[coding-tab] /auth/me failed\", err);\n state.me = null;\n render();\n return;\n }\n try {\n const { models } = await api<{ models: ModelOption[] }>(\"/models\");\n state.models = models;\n if (models.length > 0 && !models.find((m) => m.choice === state.model)) {\n state.model = models[0]!.choice;\n }\n } catch (err) {\n console.error(\"[coding-tab] /models failed\", err);\n state.models = [\n { choice: \"sonnet\", cursorModelId: \"auto\", displayName: \"Sonnet (fallback)\" },\n { choice: \"opus\", cursorModelId: \"auto\", displayName: \"Opus (fallback)\" },\n ];\n }\n await loadChatList();\n if (state.chats.length > 0 && !state.activeChatId) {\n state.activeChatId = state.chats[0]!.meta.id;\n ensureChatLoaded(state.activeChatId).then(() => render());\n }\n render();\n }\n\n bootstrap();\n\n return {\n destroy() {\n for (const c of state.chats) c.abort?.abort();\n el.removeChild(root);\n },\n };\n}\n\nif (typeof window !== \"undefined\") {\n (window as unknown as { CodingTab?: { mountCodingTab: typeof mountCodingTab } }).CodingTab = {\n mountCodingTab,\n };\n}\n","/**\n * Tiny zero-dependency markdown renderer used for Plan-mode assistant turns.\n *\n * Supports headings (`#`/`##`/`###`/`####`), unordered (`-`/`*`) and ordered\n * (`1.`) lists, fenced code blocks, inline `code`, `**bold**`, `*italic*`,\n * `[link](url)`, horizontal rules (`---`), paragraphs, and hard breaks.\n *\n * All text is HTML-escaped before any markdown substitutions are applied so\n * the output is safe to inject via `innerHTML` even when the upstream model\n * dumps unsanitised user input back to us.\n */\n\nexport function escapeHtml(s: string): string {\n return s.replace(/[&<>\"']/g, (c) =>\n (\n {\n \"&\": \"&\",\n \"<\": \"<\",\n \">\": \">\",\n '\"': \""\",\n \"'\": \"'\",\n } as Record<string, string>\n )[c]!,\n );\n}\n\nfunction applyInline(text: string): string {\n // Inline code first — anything inside backticks should be inert.\n const segments: string[] = [];\n const codeRe = /`([^`\\n]+)`/g;\n let lastIndex = 0;\n let m: RegExpExecArray | null;\n while ((m = codeRe.exec(text)) !== null) {\n segments.push(applyInlineRest(text.slice(lastIndex, m.index)));\n segments.push(`<code>${m[1]!}</code>`);\n lastIndex = m.index + m[0].length;\n }\n segments.push(applyInlineRest(text.slice(lastIndex)));\n return segments.join(\"\");\n}\n\nfunction applyInlineRest(text: string): string {\n return text\n .replace(/\\*\\*([^*\\n]+)\\*\\*/g, \"<strong>$1</strong>\")\n .replace(/(^|[^*])\\*([^*\\n]+)\\*(?!\\*)/g, \"$1<em>$2</em>\")\n .replace(/\\[([^\\]\\n]+)\\]\\(([^)\\s]+)\\)/g, (_match, label: string, url: string) => {\n const safeUrl = /^(https?:|mailto:|\\/|#)/i.test(url) ? url : \"#\";\n return `<a href=\"${safeUrl}\" target=\"_blank\" rel=\"noreferrer noopener\">${label}</a>`;\n });\n}\n\ninterface ListContext {\n type: \"ul\" | \"ol\";\n indent: number;\n items: string[];\n}\n\nfunction flushParagraph(buf: string[], out: string[]) {\n if (buf.length === 0) return;\n const joined = buf.join(\" \").trim();\n buf.length = 0;\n if (!joined) return;\n out.push(`<p>${applyInline(joined)}</p>`);\n}\n\nfunction flushList(stack: ListContext[], out: string[]) {\n while (stack.length > 0) {\n const ctx = stack.pop()!;\n out.push(`<${ctx.type}>${ctx.items.join(\"\")}</${ctx.type}>`);\n }\n}\n\nexport function renderMarkdown(input: string): string {\n // Escape first so list markers / fences are still recognised but any HTML in\n // the source becomes inert.\n const escaped = escapeHtml(input).replace(/\\r\\n/g, \"\\n\");\n const lines = escaped.split(\"\\n\");\n\n const out: string[] = [];\n const paragraph: string[] = [];\n const listStack: ListContext[] = [];\n let inCode = false;\n let codeLang = \"\";\n const codeBuf: string[] = [];\n\n const closeOpenBlocks = () => {\n flushParagraph(paragraph, out);\n flushList(listStack, out);\n };\n\n for (let i = 0; i < lines.length; i++) {\n const line = lines[i]!;\n\n // Fenced code blocks\n const fence = line.match(/^\\s*```(\\w*)\\s*$/);\n if (fence) {\n if (inCode) {\n const cls = codeLang ? ` class=\"lang-${codeLang}\"` : \"\";\n out.push(`<pre><code${cls}>${codeBuf.join(\"\\n\")}</code></pre>`);\n codeBuf.length = 0;\n codeLang = \"\";\n inCode = false;\n } else {\n closeOpenBlocks();\n inCode = true;\n codeLang = fence[1] ?? \"\";\n }\n continue;\n }\n if (inCode) {\n codeBuf.push(line);\n continue;\n }\n\n // Blank line: paragraph break, list break.\n if (!line.trim()) {\n closeOpenBlocks();\n continue;\n }\n\n // Horizontal rule.\n if (/^\\s*(-{3,}|\\*{3,}|_{3,})\\s*$/.test(line)) {\n closeOpenBlocks();\n out.push(\"<hr />\");\n continue;\n }\n\n // Headings.\n const h = line.match(/^(#{1,6})\\s+(.*)$/);\n if (h) {\n closeOpenBlocks();\n const level = Math.min(6, h[1]!.length);\n out.push(`<h${level}>${applyInline(h[2]!.trim())}</h${level}>`);\n continue;\n }\n\n // List items.\n const ul = line.match(/^(\\s*)[-*]\\s+(.*)$/);\n const ol = line.match(/^(\\s*)(\\d+)\\.\\s+(.*)$/);\n if (ul || ol) {\n flushParagraph(paragraph, out);\n const indent = (ul ? ul[1] : ol![1])!.length;\n const type: \"ul\" | \"ol\" = ul ? \"ul\" : \"ol\";\n const content = (ul ? ul[2] : ol![3])!;\n // Pop deeper lists.\n while (listStack.length > 0 && listStack[listStack.length - 1]!.indent > indent) {\n const ctx = listStack.pop()!;\n const prev = listStack[listStack.length - 1]?.items;\n const html = `<${ctx.type}>${ctx.items.join(\"\")}</${ctx.type}>`;\n if (prev && prev.length > 0) {\n prev[prev.length - 1] = prev[prev.length - 1]!.replace(\n /<\\/li>$/,\n `${html}</li>`,\n );\n } else {\n out.push(html);\n }\n }\n const top = listStack[listStack.length - 1];\n if (!top || top.indent < indent || top.type !== type) {\n listStack.push({ type, indent, items: [`<li>${applyInline(content)}</li>`] });\n } else {\n top.items.push(`<li>${applyInline(content)}</li>`);\n }\n continue;\n }\n\n // If we were in a list and hit a non-list line, close the list.\n if (listStack.length > 0) {\n flushList(listStack, out);\n }\n paragraph.push(line.trim());\n }\n\n if (inCode) {\n out.push(`<pre><code>${codeBuf.join(\"\\n\")}</code></pre>`);\n } else {\n closeOpenBlocks();\n }\n\n return out.join(\"\\n\");\n}\n"],"mappings":"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","names":["tab_exports","__export","mountCodingTab","escapeHtml","s","c","applyInline","text","segments","codeRe","lastIndex","m","applyInlineRest","_match","label","url","flushParagraph","buf","out","joined","flushList","stack","ctx","renderMarkdown","input","lines","paragraph","listStack","inCode","codeLang","codeBuf","closeOpenBlocks","i","line","fence","cls","h","level","ul","ol","indent","type","content","prev","html","top","ICON_GITHUB","ICON_PLUS","ICON_TRASH","ICON_PENCIL","ICON_MENU","ICON_CHEVRON","escapeAttr","s","escapeHtml","repoSlug","url","m","repoWebUrl","githubPullsUrl","cryptoRandomId","relativeTime","ts","diff","mountCodingTab","el","options","apiBase","root","stylesheetInjected","injectStylesheet","link","state","activeChat","c","api","path","init","resp","loadChatList","chats","existing","meta","prev","err","ensureChatLoaded","chatId","t","chat","createChat","thread","switchChat","render","deleteChat","renameChat","next","trimmed","headerHtml","sidebarHtml","renderChatRow","paneHtml","renderChatPane","renderEmptyPane","bindEvents","active","streaming","turns","threadHtml","renderTurn","placeholder","turn","text","extractText","eventsHtml","evt","renderEvent","statusBadge","planActions","prHtml","renderPr","renderBranchFallback","isPlan","renderMarkdown","renderInlineText","renderToolEvent","para","_m","label","lead","p","summary","summarizeTool","expanded","detailHtml","renderToolDetail","name","args","a","lower","pattern","lines","argsBlock","formatBlob","resultBlock","value","branch","slug","pr","merge","mergeButton","statusLine","e","b","id","onSend","ev","onCancel","onExecute","onMerge","promptEl","prompt","userTurn","assistantTurn","streamSse","appendErrorToTurn","prUrl","result","message","body","reader","decoder","buffer","lastWasText","done","events","raw","dataLine","l","applyStreamEvent","last","bootstrap","me","status","models"]}
|
package/dist/server.cjs
CHANGED
|
@@ -551,6 +551,18 @@ function parsePrUrl(url) {
|
|
|
551
551
|
if (!match) return void 0;
|
|
552
552
|
return { url, owner: match[1], repo: match[2], number: Number(match[3]) };
|
|
553
553
|
}
|
|
554
|
+
function repoWebUrl(repoUrl) {
|
|
555
|
+
const m = repoUrl.match(/github\.com[/:]([^/]+)\/([^/]+?)(?:\.git)?\/?$/);
|
|
556
|
+
return m ? `https://github.com/${m[1]}/${m[2]}` : repoUrl.replace(/\.git\/?$/, "");
|
|
557
|
+
}
|
|
558
|
+
function buildBranchInfo(repoUrl, branch) {
|
|
559
|
+
const web = repoWebUrl(repoUrl);
|
|
560
|
+
return {
|
|
561
|
+
repoUrl: web,
|
|
562
|
+
branch,
|
|
563
|
+
compareUrl: `${web}/pull/new/${encodeURIComponent(branch)}`
|
|
564
|
+
};
|
|
565
|
+
}
|
|
554
566
|
function modeInstructionPrefix(mode, prompt) {
|
|
555
567
|
return mode === "plan" ? `${PLAN_INSTRUCTION}
|
|
556
568
|
|
|
@@ -628,12 +640,13 @@ async function streamRun(res, run, ctx) {
|
|
|
628
640
|
const flushDebounceMs = 750;
|
|
629
641
|
let flushPending = false;
|
|
630
642
|
let lastFlush = 0;
|
|
631
|
-
const persist = async (status, pr) => {
|
|
643
|
+
const persist = async (status, pr, branch) => {
|
|
632
644
|
try {
|
|
633
645
|
await ctx.storage.patchTurn(ctx.chat.id, ctx.turn.id, {
|
|
634
646
|
events: ctx.buffer.snapshot(),
|
|
635
647
|
...status ? { status } : {},
|
|
636
|
-
...pr ? { pr } : {}
|
|
648
|
+
...pr ? { pr } : {},
|
|
649
|
+
...branch ? { branch } : {}
|
|
637
650
|
});
|
|
638
651
|
lastFlush = Date.now();
|
|
639
652
|
} catch (err) {
|
|
@@ -652,6 +665,7 @@ async function streamRun(res, run, ctx) {
|
|
|
652
665
|
};
|
|
653
666
|
let finalStatus = "finished";
|
|
654
667
|
let finalPr;
|
|
668
|
+
let finalBranch;
|
|
655
669
|
try {
|
|
656
670
|
for await (const message of run.stream()) {
|
|
657
671
|
if (res.writableEnded) break;
|
|
@@ -702,13 +716,20 @@ async function streamRun(res, run, ctx) {
|
|
|
702
716
|
scheduleFlush();
|
|
703
717
|
}
|
|
704
718
|
const result = await run.wait();
|
|
705
|
-
const
|
|
706
|
-
finalPr = parsePrUrl(prUrl);
|
|
719
|
+
const branchWithPr = result.git?.branches?.find((b) => b.prUrl);
|
|
720
|
+
finalPr = parsePrUrl(branchWithPr?.prUrl);
|
|
721
|
+
if (!finalPr) {
|
|
722
|
+
const firstBranch = result.git?.branches?.find((b) => b.branch);
|
|
723
|
+
if (firstBranch?.branch) {
|
|
724
|
+
finalBranch = buildBranchInfo(firstBranch.repoUrl, firstBranch.branch);
|
|
725
|
+
}
|
|
726
|
+
}
|
|
707
727
|
finalStatus = result.status ?? "finished";
|
|
708
728
|
sse(res, {
|
|
709
729
|
kind: "result",
|
|
710
730
|
status: result.status,
|
|
711
731
|
pr: finalPr,
|
|
732
|
+
branch: finalBranch,
|
|
712
733
|
durationMs: result.durationMs
|
|
713
734
|
});
|
|
714
735
|
} catch (err) {
|
|
@@ -717,7 +738,7 @@ async function streamRun(res, run, ctx) {
|
|
|
717
738
|
const retryable = err instanceof import_sdk2.CursorAgentError ? Boolean(err.isRetryable) : false;
|
|
718
739
|
sse(res, { kind: "error", message, retryable });
|
|
719
740
|
} finally {
|
|
720
|
-
await persist(finalStatus, finalPr);
|
|
741
|
+
await persist(finalStatus, finalPr, finalBranch);
|
|
721
742
|
if (!res.writableEnded) res.end();
|
|
722
743
|
}
|
|
723
744
|
}
|