@agent-native/core 0.24.1 → 0.24.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.
Files changed (58) hide show
  1. package/README.md +22 -2
  2. package/dist/agent/thread-data-builder.d.ts +1 -0
  3. package/dist/agent/thread-data-builder.d.ts.map +1 -1
  4. package/dist/agent/thread-data-builder.js +8 -2
  5. package/dist/agent/thread-data-builder.js.map +1 -1
  6. package/dist/chat-threads/store.d.ts +9 -0
  7. package/dist/chat-threads/store.d.ts.map +1 -1
  8. package/dist/chat-threads/store.js +73 -8
  9. package/dist/chat-threads/store.js.map +1 -1
  10. package/dist/cli/templates-meta.js +1 -1
  11. package/dist/cli/templates-meta.js.map +1 -1
  12. package/dist/client/AgentPanel.d.ts +2 -0
  13. package/dist/client/AgentPanel.d.ts.map +1 -1
  14. package/dist/client/AgentPanel.js +2 -2
  15. package/dist/client/AgentPanel.js.map +1 -1
  16. package/dist/client/MultiTabAssistantChat.d.ts +3 -1
  17. package/dist/client/MultiTabAssistantChat.d.ts.map +1 -1
  18. package/dist/client/MultiTabAssistantChat.js +12 -3
  19. package/dist/client/MultiTabAssistantChat.js.map +1 -1
  20. package/dist/client/NewWorkspaceAppFlow.js +2 -2
  21. package/dist/client/NewWorkspaceAppFlow.js.map +1 -1
  22. package/dist/client/onboarding/OnboardingPanel.js +1 -1
  23. package/dist/client/onboarding/OnboardingPanel.js.map +1 -1
  24. package/dist/client/org/OrgSwitcher.d.ts.map +1 -1
  25. package/dist/client/org/OrgSwitcher.js +29 -2
  26. package/dist/client/org/OrgSwitcher.js.map +1 -1
  27. package/dist/client/org/workspace-app-links.d.ts +1 -0
  28. package/dist/client/org/workspace-app-links.d.ts.map +1 -1
  29. package/dist/client/org/workspace-app-links.js +7 -1
  30. package/dist/client/org/workspace-app-links.js.map +1 -1
  31. package/dist/client/use-chat-threads.d.ts +9 -0
  32. package/dist/client/use-chat-threads.d.ts.map +1 -1
  33. package/dist/client/use-chat-threads.js +135 -32
  34. package/dist/client/use-chat-threads.js.map +1 -1
  35. package/dist/client/use-chat-threads.spec.js +143 -0
  36. package/dist/client/use-chat-threads.spec.js.map +1 -1
  37. package/dist/scripts/chat/search-chats.d.ts.map +1 -1
  38. package/dist/scripts/chat/search-chats.js +7 -1
  39. package/dist/scripts/chat/search-chats.js.map +1 -1
  40. package/dist/server/agent-chat-plugin.d.ts.map +1 -1
  41. package/dist/server/agent-chat-plugin.js +115 -6
  42. package/dist/server/agent-chat-plugin.js.map +1 -1
  43. package/dist/server/framework-request-handler.d.ts.map +1 -1
  44. package/dist/server/framework-request-handler.js +45 -0
  45. package/dist/server/framework-request-handler.js.map +1 -1
  46. package/dist/server/index.d.ts +1 -1
  47. package/dist/server/index.d.ts.map +1 -1
  48. package/dist/server/index.js +1 -1
  49. package/dist/server/index.js.map +1 -1
  50. package/dist/templates/workspace-root/AGENTS.md +6 -5
  51. package/dist/templates/workspace-root/README.md +1 -1
  52. package/docs/content/cloneable-saas.md +1 -0
  53. package/docs/content/faq.md +1 -0
  54. package/docs/content/getting-started.md +1 -0
  55. package/docs/content/multi-tenancy.md +1 -1
  56. package/package.json +1 -1
  57. package/src/templates/workspace-root/AGENTS.md +6 -5
  58. package/src/templates/workspace-root/README.md +1 -1
@@ -97,6 +97,44 @@ describe("useChatThreads", () => {
97
97
  "old-project-thread",
98
98
  ]);
99
99
  });
100
+ it("can ignore a saved active thread and start fresh immediately", async () => {
101
+ window.localStorage.setItem("agent-chat-active-thread:brain", "old-brain-thread");
102
+ const oldThread = {
103
+ id: "old-brain-thread",
104
+ title: "Using the Brain demo corpus",
105
+ preview: "what should the demo cite?",
106
+ messageCount: 2,
107
+ createdAt: 1,
108
+ updatedAt: 2,
109
+ scope: null,
110
+ };
111
+ const fetchMock = vi.fn(async (url, init) => {
112
+ if (url === "/chat/threads" && !init) {
113
+ return jsonResponse({ threads: [oldThread] });
114
+ }
115
+ throw new Error(`Unexpected fetch: ${url}`);
116
+ });
117
+ vi.stubGlobal("fetch", fetchMock);
118
+ let hook = null;
119
+ function Harness() {
120
+ hook = useChatThreads("/chat", "brain", null, {
121
+ restoreActiveThread: false,
122
+ });
123
+ return null;
124
+ }
125
+ await act(async () => {
126
+ root.render(_jsx(Harness, {}));
127
+ });
128
+ expect(hook.activeThreadId).toBe("forked-thread");
129
+ await act(async () => {
130
+ await Promise.resolve();
131
+ await Promise.resolve();
132
+ });
133
+ expect(hook.threads.map((thread) => thread.id)).toEqual([
134
+ "forked-thread",
135
+ "old-brain-thread",
136
+ ]);
137
+ });
100
138
  it("keeps the active general chat visible when entering a scoped surface", async () => {
101
139
  window.localStorage.setItem("agent-chat-active-thread:forms-app", "general-thread");
102
140
  const generalThread = {
@@ -388,5 +426,110 @@ describe("useChatThreads", () => {
388
426
  messageCount: 2,
389
427
  });
390
428
  });
429
+ it("renames a thread optimistically", async () => {
430
+ const sourceThread = {
431
+ id: "thread-1",
432
+ title: "Old title",
433
+ preview: "old preview",
434
+ messageCount: 1,
435
+ createdAt: 1,
436
+ updatedAt: 2,
437
+ scope: null,
438
+ };
439
+ const fetchMock = vi.fn(async (url, init) => {
440
+ if (url === "/chat/threads" && !init) {
441
+ return jsonResponse({ threads: [sourceThread] });
442
+ }
443
+ if (url === "/chat/threads/thread-1/rename" && init?.method === "POST") {
444
+ return jsonResponse({ ok: true });
445
+ }
446
+ throw new Error(`Unexpected fetch: ${url}`);
447
+ });
448
+ vi.stubGlobal("fetch", fetchMock);
449
+ let hook = null;
450
+ function Harness() {
451
+ hook = useChatThreads("/chat", "rename-test", null, {
452
+ autoCreate: false,
453
+ });
454
+ return null;
455
+ }
456
+ await act(async () => {
457
+ root.render(_jsx(Harness, {}));
458
+ });
459
+ await act(async () => {
460
+ await Promise.resolve();
461
+ await Promise.resolve();
462
+ });
463
+ await act(async () => {
464
+ await hook.renameThread("thread-1", " New title ");
465
+ });
466
+ expect(JSON.parse(fetchMock.mock.calls[1][1].body)).toEqual({
467
+ title: "New title",
468
+ });
469
+ expect(hook.threads.find((thread) => thread.id === "thread-1")?.title).toBe("New title");
470
+ });
471
+ it("preserves a user rename over generated titles and later saves", async () => {
472
+ const sourceThread = {
473
+ id: "thread-1",
474
+ title: "Old title",
475
+ preview: "old preview",
476
+ messageCount: 1,
477
+ createdAt: 1,
478
+ updatedAt: 2,
479
+ scope: null,
480
+ };
481
+ let resolveGenerate = null;
482
+ const fetchMock = vi.fn(async (url, init) => {
483
+ if (url === "/chat/threads" && !init) {
484
+ return jsonResponse({ threads: [sourceThread] });
485
+ }
486
+ if (url === "/chat/generate-title" && init?.method === "POST") {
487
+ return new Promise((resolve) => {
488
+ resolveGenerate = resolve;
489
+ });
490
+ }
491
+ if (url === "/chat/threads/thread-1/rename" && init?.method === "POST") {
492
+ return jsonResponse({ ok: true });
493
+ }
494
+ if (url === "/chat/threads/thread-1" && init?.method === "PUT") {
495
+ return jsonResponse({ ok: true });
496
+ }
497
+ throw new Error(`Unexpected fetch: ${url}`);
498
+ });
499
+ vi.stubGlobal("fetch", fetchMock);
500
+ let hook = null;
501
+ function Harness() {
502
+ hook = useChatThreads("/chat", "rename-generated-test", null, {
503
+ autoCreate: false,
504
+ });
505
+ return null;
506
+ }
507
+ await act(async () => {
508
+ root.render(_jsx(Harness, {}));
509
+ });
510
+ await act(async () => {
511
+ await Promise.resolve();
512
+ await Promise.resolve();
513
+ });
514
+ const generatedTitlePromise = hook.generateTitle("thread-1", "Please summarize this chat");
515
+ await act(async () => {
516
+ await hook.renameThread("thread-1", "User title");
517
+ });
518
+ await act(async () => {
519
+ resolveGenerate(jsonResponse({ title: "Generated title" }));
520
+ await generatedTitlePromise;
521
+ });
522
+ await act(async () => {
523
+ await hook.saveThreadData("thread-1", {
524
+ threadData: "",
525
+ title: "Generated title",
526
+ preview: "Please summarize this chat",
527
+ titleSource: "generated",
528
+ });
529
+ });
530
+ const saveCall = fetchMock.mock.calls.find(([url, init]) => url === "/chat/threads/thread-1" && init?.method === "PUT");
531
+ expect(JSON.parse(saveCall[1].body).title).toBe("User title");
532
+ expect(hook.threads.find((thread) => thread.id === "thread-1")?.title).toBe("User title");
533
+ });
391
534
  });
392
535
  //# sourceMappingURL=use-chat-threads.spec.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"use-chat-threads.spec.js","sourceRoot":"","sources":["../../src/client/use-chat-threads.spec.tsx"],"names":[],"mappings":"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","sourcesContent":["// @vitest-environment happy-dom\n\nimport React, { act } from \"react\";\nimport { createRoot, type Root } from \"react-dom/client\";\nimport { afterEach, beforeEach, describe, expect, it, vi } from \"vitest\";\nimport {\n useChatThreads,\n type ChatThreadScope,\n type ChatThreadSnapshot,\n type ChatThreadSummary,\n} from \"./use-chat-threads.js\";\n\nfunction jsonResponse(data: unknown) {\n return new Response(JSON.stringify(data), {\n headers: { \"Content-Type\": \"application/json\" },\n });\n}\n\ndescribe(\"useChatThreads\", () => {\n let container: HTMLDivElement;\n let root: Root;\n\n beforeEach(() => {\n vi.stubGlobal(\"IS_REACT_ACT_ENVIRONMENT\", true);\n vi.stubGlobal(\"crypto\", { randomUUID: () => \"forked-thread\" });\n window.localStorage.clear();\n container = document.createElement(\"div\");\n document.body.appendChild(container);\n root = createRoot(container);\n });\n\n afterEach(() => {\n act(() => {\n root.unmount();\n });\n container.remove();\n vi.unstubAllGlobals();\n });\n\n it(\"starts fresh when no active thread is saved, even if server history exists\", async () => {\n const oldThread: ChatThreadSummary = {\n id: \"old-project-thread\",\n title: \"Animated charting tool\",\n preview: \"make the chart more playful\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [oldThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"analytics-project\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"forked-thread\");\n expect(hook!.threads.map((thread) => thread.id)).toEqual([\n \"forked-thread\",\n \"old-project-thread\",\n ]);\n });\n\n it(\"keeps a saved active thread when it still exists on the server\", async () => {\n window.localStorage.setItem(\n \"agent-chat-active-thread:analytics-project\",\n \"old-project-thread\",\n );\n const oldThread: ChatThreadSummary = {\n id: \"old-project-thread\",\n title: \"Analytics for Academy\",\n preview: \"show weekly signups\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [oldThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"analytics-project\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"old-project-thread\");\n expect(hook!.threads.map((thread) => thread.id)).toEqual([\n \"old-project-thread\",\n ]);\n });\n\n it(\"keeps the active general chat visible when entering a scoped surface\", async () => {\n window.localStorage.setItem(\n \"agent-chat-active-thread:forms-app\",\n \"general-thread\",\n );\n const generalThread: ChatThreadSummary = {\n id: \"general-thread\",\n title: \"Create a form\",\n preview: \"make me a form\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const formThread: ChatThreadSummary = {\n id: \"form-thread\",\n title: \"Form edits\",\n preview: \"add another question\",\n messageCount: 2,\n createdAt: 3,\n updatedAt: 4,\n scope: { type: \"form\", id: \"form-1\", label: \"Hackathon\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [generalThread, formThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness({ scope }: { scope?: ChatThreadScope | null }) {\n hook = useChatThreads(\"/chat\", \"forms-app\", scope);\n return null;\n }\n\n await act(async () => {\n root.render(<Harness scope={null} />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"general-thread\");\n\n await act(async () => {\n root.render(\n <Harness scope={{ type: \"form\", id: \"form-1\", label: \"Hackathon\" }} />,\n );\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"general-thread\");\n expect(\n window.localStorage.getItem(\n \"agent-chat-active-thread:forms-app:scope:form:form-1\",\n ),\n ).toBeNull();\n expect(\n window.localStorage.getItem(\"agent-chat-active-thread:forms-app\"),\n ).toBe(\"general-thread\");\n });\n\n it(\"switches back to the general chat when leaving a scoped thread\", async () => {\n window.localStorage.setItem(\n \"agent-chat-active-thread:forms-app\",\n \"general-thread\",\n );\n window.localStorage.setItem(\n \"agent-chat-active-thread:forms-app:scope:form:form-1\",\n \"form-thread\",\n );\n const generalThread: ChatThreadSummary = {\n id: \"general-thread\",\n title: \"Create a form\",\n preview: \"make me a form\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const formThread: ChatThreadSummary = {\n id: \"form-thread\",\n title: \"Form edits\",\n preview: \"add another question\",\n messageCount: 2,\n createdAt: 3,\n updatedAt: 4,\n scope: { type: \"form\", id: \"form-1\", label: \"Hackathon\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [formThread, generalThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness({ scope }: { scope?: ChatThreadScope | null }) {\n hook = useChatThreads(\"/chat\", \"forms-app\", scope);\n return null;\n }\n\n await act(async () => {\n root.render(\n <Harness scope={{ type: \"form\", id: \"form-1\", label: \"Hackathon\" }} />,\n );\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"form-thread\");\n\n await act(async () => {\n root.render(<Harness scope={null} />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"general-thread\");\n });\n\n it(\"sends the current client snapshot when forking a thread\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"source-thread\",\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: { type: \"dashboard\", id: \"dash-1\", label: \"Pipeline\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/threads/source-thread/fork\") {\n return jsonResponse({\n ...sourceThread,\n id: \"forked-thread\",\n title: \"Pipeline (fork)\",\n });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"fork-test\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n const snapshot: ChatThreadSnapshot = {\n threadData: JSON.stringify({ messages: [{ message: { id: \"m1\" } }] }),\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 1,\n };\n\n let forkedId: string | null = null;\n await act(async () => {\n forkedId = await hook!.forkThread(\"source-thread\", snapshot);\n });\n\n expect(forkedId).toBe(\"forked-thread\");\n const forkCall = fetchMock.mock.calls.find(\n ([url]) => url === \"/chat/threads/source-thread/fork\",\n );\n expect(forkCall).toBeDefined();\n expect(JSON.parse(forkCall![1]!.body as string)).toEqual({\n id: \"forked-thread\",\n source: { ...snapshot, scope: sourceThread.scope },\n });\n });\n\n it(\"creates a fork from the client snapshot when the fork endpoint cannot find the source\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"source-thread\",\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: { type: \"deck\", id: \"deck-1\", label: \"Pipeline deck\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/threads/source-thread/fork\") {\n return new Response(JSON.stringify({ error: \"Thread not found\" }), {\n status: 404,\n headers: { \"Content-Type\": \"application/json\" },\n });\n }\n if (url === \"/chat/threads\" && init?.method === \"POST\") {\n return jsonResponse({\n id: \"forked-thread\",\n title: \"Pipeline (fork)\",\n preview: \"\",\n messageCount: 0,\n createdAt: 3,\n updatedAt: 3,\n scope: sourceThread.scope,\n });\n }\n if (url === \"/chat/threads/forked-thread\" && init?.method === \"PUT\") {\n return jsonResponse({ ok: true });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"fork-test\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n const snapshot: ChatThreadSnapshot = {\n threadData: JSON.stringify({ messages: [{ message: { id: \"m1\" } }] }),\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 1,\n };\n\n let forkedId: string | null = null;\n await act(async () => {\n forkedId = await hook!.forkThread(\"source-thread\", snapshot);\n });\n\n expect(forkedId).toBe(\"forked-thread\");\n const createCall = fetchMock.mock.calls.find(\n ([url, init]) => url === \"/chat/threads\" && init?.method === \"POST\",\n );\n expect(createCall).toBeDefined();\n expect(JSON.parse(createCall![1]!.body as string)).toEqual({\n id: \"forked-thread\",\n title: \"Pipeline (fork)\",\n scope: sourceThread.scope,\n });\n const saveCall = fetchMock.mock.calls.find(\n ([url, init]) =>\n url === \"/chat/threads/forked-thread\" && init?.method === \"PUT\",\n );\n expect(saveCall).toBeDefined();\n expect(JSON.parse(saveCall![1]!.body as string)).toEqual({\n threadData: snapshot.threadData,\n title: \"Pipeline (fork)\",\n preview: snapshot.preview,\n messageCount: snapshot.messageCount,\n scope: sourceThread.scope,\n });\n });\n\n it(\"keeps generated titles when later thread saves update the preview\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"thread-1\",\n title: \"Using the Brain demo data for this example\",\n preview: \"Using the Brain demo data for this example\",\n messageCount: 1,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/threads/thread-1\" && init?.method === \"PUT\") {\n return jsonResponse({ ok: true });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"title-test\", null, {\n autoCreate: false,\n });\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n await act(async () => {\n await hook!.saveThreadData(\"thread-1\", {\n threadData: \"\",\n title: \"Brain Demo Setup\",\n preview: \"Using the Brain demo data for this example\",\n titleSource: \"generated\",\n });\n });\n\n await act(async () => {\n await hook!.saveThreadData(\"thread-1\", {\n threadData: JSON.stringify({ messages: [] }),\n title: \"Using the Brain demo data for this example\",\n preview: \"What should the demo answer cite?\",\n messageCount: 2,\n });\n });\n\n const saveCalls = fetchMock.mock.calls.filter(\n ([url, init]) =>\n url === \"/chat/threads/thread-1\" && init?.method === \"PUT\",\n );\n expect(JSON.parse(saveCalls[0]![1]!.body as string).title).toBe(\n \"Brain Demo Setup\",\n );\n expect(JSON.parse(saveCalls[1]![1]!.body as string).title).toBe(\n \"Brain Demo Setup\",\n );\n expect(\n hook!.threads.find((thread) => thread.id === \"thread-1\"),\n ).toMatchObject({\n title: \"Brain Demo Setup\",\n preview: \"What should the demo answer cite?\",\n messageCount: 2,\n });\n });\n});\n"]}
1
+ {"version":3,"file":"use-chat-threads.spec.js","sourceRoot":"","sources":["../../src/client/use-chat-threads.spec.tsx"],"names":[],"mappings":"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","sourcesContent":["// @vitest-environment happy-dom\n\nimport React, { act } from \"react\";\nimport { createRoot, type Root } from \"react-dom/client\";\nimport { afterEach, beforeEach, describe, expect, it, vi } from \"vitest\";\nimport {\n useChatThreads,\n type ChatThreadScope,\n type ChatThreadSnapshot,\n type ChatThreadSummary,\n} from \"./use-chat-threads.js\";\n\nfunction jsonResponse(data: unknown) {\n return new Response(JSON.stringify(data), {\n headers: { \"Content-Type\": \"application/json\" },\n });\n}\n\ndescribe(\"useChatThreads\", () => {\n let container: HTMLDivElement;\n let root: Root;\n\n beforeEach(() => {\n vi.stubGlobal(\"IS_REACT_ACT_ENVIRONMENT\", true);\n vi.stubGlobal(\"crypto\", { randomUUID: () => \"forked-thread\" });\n window.localStorage.clear();\n container = document.createElement(\"div\");\n document.body.appendChild(container);\n root = createRoot(container);\n });\n\n afterEach(() => {\n act(() => {\n root.unmount();\n });\n container.remove();\n vi.unstubAllGlobals();\n });\n\n it(\"starts fresh when no active thread is saved, even if server history exists\", async () => {\n const oldThread: ChatThreadSummary = {\n id: \"old-project-thread\",\n title: \"Animated charting tool\",\n preview: \"make the chart more playful\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [oldThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"analytics-project\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"forked-thread\");\n expect(hook!.threads.map((thread) => thread.id)).toEqual([\n \"forked-thread\",\n \"old-project-thread\",\n ]);\n });\n\n it(\"keeps a saved active thread when it still exists on the server\", async () => {\n window.localStorage.setItem(\n \"agent-chat-active-thread:analytics-project\",\n \"old-project-thread\",\n );\n const oldThread: ChatThreadSummary = {\n id: \"old-project-thread\",\n title: \"Analytics for Academy\",\n preview: \"show weekly signups\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [oldThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"analytics-project\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"old-project-thread\");\n expect(hook!.threads.map((thread) => thread.id)).toEqual([\n \"old-project-thread\",\n ]);\n });\n\n it(\"can ignore a saved active thread and start fresh immediately\", async () => {\n window.localStorage.setItem(\n \"agent-chat-active-thread:brain\",\n \"old-brain-thread\",\n );\n const oldThread: ChatThreadSummary = {\n id: \"old-brain-thread\",\n title: \"Using the Brain demo corpus\",\n preview: \"what should the demo cite?\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [oldThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"brain\", null, {\n restoreActiveThread: false,\n });\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n\n expect(hook!.activeThreadId).toBe(\"forked-thread\");\n\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.threads.map((thread) => thread.id)).toEqual([\n \"forked-thread\",\n \"old-brain-thread\",\n ]);\n });\n\n it(\"keeps the active general chat visible when entering a scoped surface\", async () => {\n window.localStorage.setItem(\n \"agent-chat-active-thread:forms-app\",\n \"general-thread\",\n );\n const generalThread: ChatThreadSummary = {\n id: \"general-thread\",\n title: \"Create a form\",\n preview: \"make me a form\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const formThread: ChatThreadSummary = {\n id: \"form-thread\",\n title: \"Form edits\",\n preview: \"add another question\",\n messageCount: 2,\n createdAt: 3,\n updatedAt: 4,\n scope: { type: \"form\", id: \"form-1\", label: \"Hackathon\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [generalThread, formThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness({ scope }: { scope?: ChatThreadScope | null }) {\n hook = useChatThreads(\"/chat\", \"forms-app\", scope);\n return null;\n }\n\n await act(async () => {\n root.render(<Harness scope={null} />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"general-thread\");\n\n await act(async () => {\n root.render(\n <Harness scope={{ type: \"form\", id: \"form-1\", label: \"Hackathon\" }} />,\n );\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"general-thread\");\n expect(\n window.localStorage.getItem(\n \"agent-chat-active-thread:forms-app:scope:form:form-1\",\n ),\n ).toBeNull();\n expect(\n window.localStorage.getItem(\"agent-chat-active-thread:forms-app\"),\n ).toBe(\"general-thread\");\n });\n\n it(\"switches back to the general chat when leaving a scoped thread\", async () => {\n window.localStorage.setItem(\n \"agent-chat-active-thread:forms-app\",\n \"general-thread\",\n );\n window.localStorage.setItem(\n \"agent-chat-active-thread:forms-app:scope:form:form-1\",\n \"form-thread\",\n );\n const generalThread: ChatThreadSummary = {\n id: \"general-thread\",\n title: \"Create a form\",\n preview: \"make me a form\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const formThread: ChatThreadSummary = {\n id: \"form-thread\",\n title: \"Form edits\",\n preview: \"add another question\",\n messageCount: 2,\n createdAt: 3,\n updatedAt: 4,\n scope: { type: \"form\", id: \"form-1\", label: \"Hackathon\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [formThread, generalThread] });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness({ scope }: { scope?: ChatThreadScope | null }) {\n hook = useChatThreads(\"/chat\", \"forms-app\", scope);\n return null;\n }\n\n await act(async () => {\n root.render(\n <Harness scope={{ type: \"form\", id: \"form-1\", label: \"Hackathon\" }} />,\n );\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"form-thread\");\n\n await act(async () => {\n root.render(<Harness scope={null} />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n expect(hook!.activeThreadId).toBe(\"general-thread\");\n });\n\n it(\"sends the current client snapshot when forking a thread\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"source-thread\",\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: { type: \"dashboard\", id: \"dash-1\", label: \"Pipeline\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/threads/source-thread/fork\") {\n return jsonResponse({\n ...sourceThread,\n id: \"forked-thread\",\n title: \"Pipeline (fork)\",\n });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"fork-test\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n const snapshot: ChatThreadSnapshot = {\n threadData: JSON.stringify({ messages: [{ message: { id: \"m1\" } }] }),\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 1,\n };\n\n let forkedId: string | null = null;\n await act(async () => {\n forkedId = await hook!.forkThread(\"source-thread\", snapshot);\n });\n\n expect(forkedId).toBe(\"forked-thread\");\n const forkCall = fetchMock.mock.calls.find(\n ([url]) => url === \"/chat/threads/source-thread/fork\",\n );\n expect(forkCall).toBeDefined();\n expect(JSON.parse(forkCall![1]!.body as string)).toEqual({\n id: \"forked-thread\",\n source: { ...snapshot, scope: sourceThread.scope },\n });\n });\n\n it(\"creates a fork from the client snapshot when the fork endpoint cannot find the source\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"source-thread\",\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 2,\n createdAt: 1,\n updatedAt: 2,\n scope: { type: \"deck\", id: \"deck-1\", label: \"Pipeline deck\" },\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/threads/source-thread/fork\") {\n return new Response(JSON.stringify({ error: \"Thread not found\" }), {\n status: 404,\n headers: { \"Content-Type\": \"application/json\" },\n });\n }\n if (url === \"/chat/threads\" && init?.method === \"POST\") {\n return jsonResponse({\n id: \"forked-thread\",\n title: \"Pipeline (fork)\",\n preview: \"\",\n messageCount: 0,\n createdAt: 3,\n updatedAt: 3,\n scope: sourceThread.scope,\n });\n }\n if (url === \"/chat/threads/forked-thread\" && init?.method === \"PUT\") {\n return jsonResponse({ ok: true });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"fork-test\");\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n const snapshot: ChatThreadSnapshot = {\n threadData: JSON.stringify({ messages: [{ message: { id: \"m1\" } }] }),\n title: \"Pipeline\",\n preview: \"make this slide better\",\n messageCount: 1,\n };\n\n let forkedId: string | null = null;\n await act(async () => {\n forkedId = await hook!.forkThread(\"source-thread\", snapshot);\n });\n\n expect(forkedId).toBe(\"forked-thread\");\n const createCall = fetchMock.mock.calls.find(\n ([url, init]) => url === \"/chat/threads\" && init?.method === \"POST\",\n );\n expect(createCall).toBeDefined();\n expect(JSON.parse(createCall![1]!.body as string)).toEqual({\n id: \"forked-thread\",\n title: \"Pipeline (fork)\",\n scope: sourceThread.scope,\n });\n const saveCall = fetchMock.mock.calls.find(\n ([url, init]) =>\n url === \"/chat/threads/forked-thread\" && init?.method === \"PUT\",\n );\n expect(saveCall).toBeDefined();\n expect(JSON.parse(saveCall![1]!.body as string)).toEqual({\n threadData: snapshot.threadData,\n title: \"Pipeline (fork)\",\n preview: snapshot.preview,\n messageCount: snapshot.messageCount,\n scope: sourceThread.scope,\n });\n });\n\n it(\"keeps generated titles when later thread saves update the preview\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"thread-1\",\n title: \"Using the Brain demo data for this example\",\n preview: \"Using the Brain demo data for this example\",\n messageCount: 1,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/threads/thread-1\" && init?.method === \"PUT\") {\n return jsonResponse({ ok: true });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"title-test\", null, {\n autoCreate: false,\n });\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n await act(async () => {\n await hook!.saveThreadData(\"thread-1\", {\n threadData: \"\",\n title: \"Brain Demo Setup\",\n preview: \"Using the Brain demo data for this example\",\n titleSource: \"generated\",\n });\n });\n\n await act(async () => {\n await hook!.saveThreadData(\"thread-1\", {\n threadData: JSON.stringify({ messages: [] }),\n title: \"Using the Brain demo data for this example\",\n preview: \"What should the demo answer cite?\",\n messageCount: 2,\n });\n });\n\n const saveCalls = fetchMock.mock.calls.filter(\n ([url, init]) =>\n url === \"/chat/threads/thread-1\" && init?.method === \"PUT\",\n );\n expect(JSON.parse(saveCalls[0]![1]!.body as string).title).toBe(\n \"Brain Demo Setup\",\n );\n expect(JSON.parse(saveCalls[1]![1]!.body as string).title).toBe(\n \"Brain Demo Setup\",\n );\n expect(\n hook!.threads.find((thread) => thread.id === \"thread-1\"),\n ).toMatchObject({\n title: \"Brain Demo Setup\",\n preview: \"What should the demo answer cite?\",\n messageCount: 2,\n });\n });\n\n it(\"renames a thread optimistically\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"thread-1\",\n title: \"Old title\",\n preview: \"old preview\",\n messageCount: 1,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/threads/thread-1/rename\" && init?.method === \"POST\") {\n return jsonResponse({ ok: true });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"rename-test\", null, {\n autoCreate: false,\n });\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n await act(async () => {\n await hook!.renameThread(\"thread-1\", \" New title \");\n });\n\n expect(JSON.parse(fetchMock.mock.calls[1]![1]!.body as string)).toEqual({\n title: \"New title\",\n });\n expect(\n hook!.threads.find((thread) => thread.id === \"thread-1\")?.title,\n ).toBe(\"New title\");\n });\n\n it(\"preserves a user rename over generated titles and later saves\", async () => {\n const sourceThread: ChatThreadSummary = {\n id: \"thread-1\",\n title: \"Old title\",\n preview: \"old preview\",\n messageCount: 1,\n createdAt: 1,\n updatedAt: 2,\n scope: null,\n };\n let resolveGenerate:\n | ((response: Response | PromiseLike<Response>) => void)\n | null = null;\n const fetchMock = vi.fn(async (url: string, init?: RequestInit) => {\n if (url === \"/chat/threads\" && !init) {\n return jsonResponse({ threads: [sourceThread] });\n }\n if (url === \"/chat/generate-title\" && init?.method === \"POST\") {\n return new Promise<Response>((resolve) => {\n resolveGenerate = resolve;\n });\n }\n if (url === \"/chat/threads/thread-1/rename\" && init?.method === \"POST\") {\n return jsonResponse({ ok: true });\n }\n if (url === \"/chat/threads/thread-1\" && init?.method === \"PUT\") {\n return jsonResponse({ ok: true });\n }\n throw new Error(`Unexpected fetch: ${url}`);\n });\n vi.stubGlobal(\"fetch\", fetchMock);\n\n let hook: ReturnType<typeof useChatThreads> | null = null;\n function Harness() {\n hook = useChatThreads(\"/chat\", \"rename-generated-test\", null, {\n autoCreate: false,\n });\n return null;\n }\n\n await act(async () => {\n root.render(<Harness />);\n });\n await act(async () => {\n await Promise.resolve();\n await Promise.resolve();\n });\n\n const generatedTitlePromise = hook!.generateTitle(\n \"thread-1\",\n \"Please summarize this chat\",\n );\n\n await act(async () => {\n await hook!.renameThread(\"thread-1\", \"User title\");\n });\n await act(async () => {\n resolveGenerate!(jsonResponse({ title: \"Generated title\" }));\n await generatedTitlePromise;\n });\n await act(async () => {\n await hook!.saveThreadData(\"thread-1\", {\n threadData: \"\",\n title: \"Generated title\",\n preview: \"Please summarize this chat\",\n titleSource: \"generated\",\n });\n });\n\n const saveCall = fetchMock.mock.calls.find(\n ([url, init]) =>\n url === \"/chat/threads/thread-1\" && init?.method === \"PUT\",\n );\n expect(JSON.parse(saveCall![1]!.body as string).title).toBe(\"User title\");\n expect(\n hook!.threads.find((thread) => thread.id === \"thread-1\")?.title,\n ).toBe(\"User title\");\n });\n});\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"search-chats.d.ts","sourceRoot":"","sources":["../../../src/scripts/chat/search-chats.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AA4BH,wBAA8B,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CA0EvE"}
1
+ {"version":3,"file":"search-chats.d.ts","sourceRoot":"","sources":["../../../src/scripts/chat/search-chats.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AA4BH,wBAA8B,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAkFvE"}
@@ -63,6 +63,8 @@ Examples:
63
63
  preview: t.preview,
64
64
  messageCount: t.messageCount,
65
65
  updatedAt: t.updatedAt,
66
+ pinnedAt: t.pinnedAt ?? null,
67
+ archivedAt: t.archivedAt ?? null,
66
68
  })),
67
69
  count: threads.length,
68
70
  }, null, 2));
@@ -80,8 +82,12 @@ Examples:
80
82
  const title = t.title || t.preview || "(untitled)";
81
83
  const msgs = t.messageCount === 1 ? "1 msg" : `${t.messageCount} msgs`;
82
84
  const time = formatTime(t.updatedAt);
85
+ const flags = [
86
+ t.pinnedAt ? "pinned" : null,
87
+ t.archivedAt ? "archived" : null,
88
+ ].filter(Boolean);
83
89
  console.log(` ${title}`);
84
- console.log(` ID: ${t.id} | ${msgs} | ${time}`);
90
+ console.log(` ID: ${t.id} | ${msgs} | ${time}${flags.length ? ` | ${flags.join(", ")}` : ""}`);
85
91
  if (t.preview && t.title && t.preview !== t.title) {
86
92
  console.log(` ${t.preview.slice(0, 80)}${t.preview.length > 80 ? "..." : ""}`);
87
93
  }
@@ -1 +1 @@
1
- {"version":3,"file":"search-chats.js","sourceRoot":"","sources":["../../../src/scripts/chat/search-chats.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Core script: search-chats\n *\n * Search or list past agent chat threads.\n *\n * Usage:\n * pnpm action search-chats [--query \"search term\"] [--limit N] [--format json]\n */\n\nimport { parseArgs, fail } from \"../utils.js\";\nimport { searchThreads, listThreads } from \"../../chat-threads/store.js\";\nimport { getRequestUserEmail } from \"../../server/request-context.js\";\n\nfunction getOwnerEmail(): string {\n const email = getRequestUserEmail() ?? process.env.AGENT_USER_EMAIL;\n if (!email) {\n fail(\n \"search-chats requires an authenticated user (request context or AGENT_USER_EMAIL env var).\",\n );\n }\n return email;\n}\n\nfunction formatTime(ts: number): string {\n const d = new Date(ts);\n const now = new Date();\n const diffMs = now.getTime() - d.getTime();\n const diffDays = Math.floor(diffMs / 86400000);\n if (diffDays === 0)\n return d.toLocaleTimeString([], { hour: \"numeric\", minute: \"2-digit\" });\n if (diffDays === 1) return \"Yesterday\";\n if (diffDays < 7) return d.toLocaleDateString([], { weekday: \"short\" });\n return d.toLocaleDateString([], { month: \"short\", day: \"numeric\" });\n}\n\nexport default async function searchChats(args: string[]): Promise<void> {\n const parsed = parseArgs(args);\n\n if (parsed.help === \"true\") {\n console.log(`Usage: pnpm action search-chats [options]\n\nOptions:\n --query <text> Search chats by title, preview, or content\n --limit N Max results (default: 20)\n --format json Output as JSON\n --help Show this help message\n\nExamples:\n pnpm action search-chats --query \"email setup\"\n pnpm action search-chats --limit 5\n pnpm action search-chats --format json`);\n return;\n }\n\n const owner = getOwnerEmail();\n const limit = parsed.limit ? parseInt(parsed.limit, 10) : 20;\n if (isNaN(limit) || limit < 1) fail(\"--limit must be a positive integer\");\n\n const query = parsed.query;\n const threads = query\n ? await searchThreads(owner, query, limit)\n : await listThreads(owner, limit, 0);\n\n if (parsed.format === \"json\") {\n console.log(\n JSON.stringify(\n {\n query: query ?? null,\n threads: threads.map((t) => ({\n id: t.id,\n title: t.title,\n preview: t.preview,\n messageCount: t.messageCount,\n updatedAt: t.updatedAt,\n })),\n count: threads.length,\n },\n null,\n 2,\n ),\n );\n return;\n }\n\n if (threads.length === 0) {\n console.log(query ? `No chats matching \"${query}\"` : \"No chat history\");\n return;\n }\n\n console.log(\n query\n ? `Chats matching \"${query}\" (${threads.length}):`\n : `Recent chats (${threads.length}):`,\n );\n console.log();\n\n for (const t of threads) {\n const title = t.title || t.preview || \"(untitled)\";\n const msgs = t.messageCount === 1 ? \"1 msg\" : `${t.messageCount} msgs`;\n const time = formatTime(t.updatedAt);\n console.log(` ${title}`);\n console.log(` ID: ${t.id} | ${msgs} | ${time}`);\n if (t.preview && t.title && t.preview !== t.title) {\n console.log(\n ` ${t.preview.slice(0, 80)}${t.preview.length > 80 ? \"...\" : \"\"}`,\n );\n }\n console.log();\n }\n}\n"]}
1
+ {"version":3,"file":"search-chats.js","sourceRoot":"","sources":["../../../src/scripts/chat/search-chats.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,aAAa,EAAE,WAAW,EAAE,MAAM,6BAA6B,CAAC;AACzE,OAAO,EAAE,mBAAmB,EAAE,MAAM,iCAAiC,CAAC;AAEtE,SAAS,aAAa;IACpB,MAAM,KAAK,GAAG,mBAAmB,EAAE,IAAI,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC;IACpE,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,IAAI,CACF,4FAA4F,CAC7F,CAAC;IACJ,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAS,UAAU,CAAC,EAAU;IAC5B,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC,CAAC;IACvB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;IACvB,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC;IAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,QAAQ,CAAC,CAAC;IAC/C,IAAI,QAAQ,KAAK,CAAC;QAChB,OAAO,CAAC,CAAC,kBAAkB,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC;IAC1E,IAAI,QAAQ,KAAK,CAAC;QAAE,OAAO,WAAW,CAAC;IACvC,IAAI,QAAQ,GAAG,CAAC;QAAE,OAAO,CAAC,CAAC,kBAAkB,CAAC,EAAE,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC,CAAC;IACxE,OAAO,CAAC,CAAC,kBAAkB,CAAC,EAAE,EAAE,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,CAAC;AACtE,CAAC;AAED,MAAM,CAAC,OAAO,CAAC,KAAK,UAAU,WAAW,CAAC,IAAc;IACtD,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;IAE/B,IAAI,MAAM,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;QAC3B,OAAO,CAAC,GAAG,CAAC;;;;;;;;;;;yCAWyB,CAAC,CAAC;QACvC,OAAO;IACT,CAAC;IAED,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;IAC9B,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IAC7D,IAAI,KAAK,CAAC,KAAK,CAAC,IAAI,KAAK,GAAG,CAAC;QAAE,IAAI,CAAC,oCAAoC,CAAC,CAAC;IAE1E,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC;IAC3B,MAAM,OAAO,GAAG,KAAK;QACnB,CAAC,CAAC,MAAM,aAAa,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC;QAC1C,CAAC,CAAC,MAAM,WAAW,CAAC,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;IAEvC,IAAI,MAAM,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;QAC7B,OAAO,CAAC,GAAG,CACT,IAAI,CAAC,SAAS,CACZ;YACE,KAAK,EAAE,KAAK,IAAI,IAAI;YACpB,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBAC3B,EAAE,EAAE,CAAC,CAAC,EAAE;gBACR,KAAK,EAAE,CAAC,CAAC,KAAK;gBACd,OAAO,EAAE,CAAC,CAAC,OAAO;gBAClB,YAAY,EAAE,CAAC,CAAC,YAAY;gBAC5B,SAAS,EAAE,CAAC,CAAC,SAAS;gBACtB,QAAQ,EAAE,CAAC,CAAC,QAAQ,IAAI,IAAI;gBAC5B,UAAU,EAAE,CAAC,CAAC,UAAU,IAAI,IAAI;aACjC,CAAC,CAAC;YACH,KAAK,EAAE,OAAO,CAAC,MAAM;SACtB,EACD,IAAI,EACJ,CAAC,CACF,CACF,CAAC;QACF,OAAO;IACT,CAAC;IAED,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACzB,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,sBAAsB,KAAK,GAAG,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC;QACxE,OAAO;IACT,CAAC;IAED,OAAO,CAAC,GAAG,CACT,KAAK;QACH,CAAC,CAAC,mBAAmB,KAAK,MAAM,OAAO,CAAC,MAAM,IAAI;QAClD,CAAC,CAAC,iBAAiB,OAAO,CAAC,MAAM,IAAI,CACxC,CAAC;IACF,OAAO,CAAC,GAAG,EAAE,CAAC;IAEd,KAAK,MAAM,CAAC,IAAI,OAAO,EAAE,CAAC;QACxB,MAAM,KAAK,GAAG,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,OAAO,IAAI,YAAY,CAAC;QACnD,MAAM,IAAI,GAAG,CAAC,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,YAAY,OAAO,CAAC;QACvE,MAAM,IAAI,GAAG,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;QACrC,MAAM,KAAK,GAAG;YACZ,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI;YAC5B,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI;SACjC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAClB,OAAO,CAAC,GAAG,CAAC,KAAK,KAAK,EAAE,CAAC,CAAC;QAC1B,OAAO,CAAC,GAAG,CACT,WAAW,CAAC,CAAC,EAAE,QAAQ,IAAI,QAAQ,IAAI,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CAC3F,CAAC;QACF,IAAI,CAAC,CAAC,OAAO,IAAI,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,OAAO,KAAK,CAAC,CAAC,KAAK,EAAE,CAAC;YAClD,OAAO,CAAC,GAAG,CACT,OAAO,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,MAAM,GAAG,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CACrE,CAAC;QACJ,CAAC;QACD,OAAO,CAAC,GAAG,EAAE,CAAC;IAChB,CAAC;AACH,CAAC","sourcesContent":["/**\n * Core script: search-chats\n *\n * Search or list past agent chat threads.\n *\n * Usage:\n * pnpm action search-chats [--query \"search term\"] [--limit N] [--format json]\n */\n\nimport { parseArgs, fail } from \"../utils.js\";\nimport { searchThreads, listThreads } from \"../../chat-threads/store.js\";\nimport { getRequestUserEmail } from \"../../server/request-context.js\";\n\nfunction getOwnerEmail(): string {\n const email = getRequestUserEmail() ?? process.env.AGENT_USER_EMAIL;\n if (!email) {\n fail(\n \"search-chats requires an authenticated user (request context or AGENT_USER_EMAIL env var).\",\n );\n }\n return email;\n}\n\nfunction formatTime(ts: number): string {\n const d = new Date(ts);\n const now = new Date();\n const diffMs = now.getTime() - d.getTime();\n const diffDays = Math.floor(diffMs / 86400000);\n if (diffDays === 0)\n return d.toLocaleTimeString([], { hour: \"numeric\", minute: \"2-digit\" });\n if (diffDays === 1) return \"Yesterday\";\n if (diffDays < 7) return d.toLocaleDateString([], { weekday: \"short\" });\n return d.toLocaleDateString([], { month: \"short\", day: \"numeric\" });\n}\n\nexport default async function searchChats(args: string[]): Promise<void> {\n const parsed = parseArgs(args);\n\n if (parsed.help === \"true\") {\n console.log(`Usage: pnpm action search-chats [options]\n\nOptions:\n --query <text> Search chats by title, preview, or content\n --limit N Max results (default: 20)\n --format json Output as JSON\n --help Show this help message\n\nExamples:\n pnpm action search-chats --query \"email setup\"\n pnpm action search-chats --limit 5\n pnpm action search-chats --format json`);\n return;\n }\n\n const owner = getOwnerEmail();\n const limit = parsed.limit ? parseInt(parsed.limit, 10) : 20;\n if (isNaN(limit) || limit < 1) fail(\"--limit must be a positive integer\");\n\n const query = parsed.query;\n const threads = query\n ? await searchThreads(owner, query, limit)\n : await listThreads(owner, limit, 0);\n\n if (parsed.format === \"json\") {\n console.log(\n JSON.stringify(\n {\n query: query ?? null,\n threads: threads.map((t) => ({\n id: t.id,\n title: t.title,\n preview: t.preview,\n messageCount: t.messageCount,\n updatedAt: t.updatedAt,\n pinnedAt: t.pinnedAt ?? null,\n archivedAt: t.archivedAt ?? null,\n })),\n count: threads.length,\n },\n null,\n 2,\n ),\n );\n return;\n }\n\n if (threads.length === 0) {\n console.log(query ? `No chats matching \"${query}\"` : \"No chat history\");\n return;\n }\n\n console.log(\n query\n ? `Chats matching \"${query}\" (${threads.length}):`\n : `Recent chats (${threads.length}):`,\n );\n console.log();\n\n for (const t of threads) {\n const title = t.title || t.preview || \"(untitled)\";\n const msgs = t.messageCount === 1 ? \"1 msg\" : `${t.messageCount} msgs`;\n const time = formatTime(t.updatedAt);\n const flags = [\n t.pinnedAt ? \"pinned\" : null,\n t.archivedAt ? \"archived\" : null,\n ].filter(Boolean);\n console.log(` ${title}`);\n console.log(\n ` ID: ${t.id} | ${msgs} | ${time}${flags.length ? ` | ${flags.join(\", \")}` : \"\"}`,\n );\n if (t.preview && t.title && t.preview !== t.title) {\n console.log(\n ` ${t.preview.slice(0, 80)}${t.preview.length > 80 ? \"...\" : \"\"}`,\n );\n }\n console.log();\n }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"agent-chat-plugin.d.ts","sourceRoot":"","sources":["../../src/server/agent-chat-plugin.ts"],"names":[],"mappings":"AAaA,OAAO,EASL,KAAK,WAAW,EACjB,MAAM,8BAA8B,CAAC;AAqBtC,OAAO,KAAK,EACV,mBAAmB,EACnB,cAAc,EACd,kBAAkB,EAElB,eAAe,EAEhB,MAAM,mBAAmB,CAAC;AAG3B,OAAO,EACL,gBAAgB,EAajB,MAAM,wBAAwB,CAAC;AA4DhC,OAAO,EAGL,KAAK,0BAA0B,EAC/B,KAAK,oBAAoB,EAC1B,MAAM,6BAA6B,CAAC;AA+SrC,wBAAgB,yBAAyB,CACvC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GACnC,KAAK,CAAC;IACP,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,WAAW,CAAC,aAAa,CAAC,CAAC;CACzC,CAAC,CASD;AAmBD,wBAAgB,wBAAwB,CACtC,MAAM,EAAE,SAAS,cAAc,EAAE,EACjC,WAAW,EAAE,SAAS,oBAAoB,EAAE,EAC5C,OAAO,GAAE,0BAA0B,GAAG;IAAE,KAAK,CAAC,EAAE,GAAG,CAAA;CAAO,GACzD;IAAE,YAAY,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CAO7C;AAkzCD,KAAK,cAAc,GAAG,CAAC,QAAQ,EAAE,GAAG,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;AAE9D,MAAM,WAAW,sBAAsB;IACrC,+DAA+D;IAC/D,OAAO,CAAC,EACJ,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,CAAC,MACG,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC;IAC9C,wCAAwC;IACxC,OAAO,CAAC,EACJ,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,CAAC,MACG,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC;IAC9C,mEAAmE;IACnE,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,qDAAqD;IACrD,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,qEAAqE;IACrE,KAAK,CAAC,EAAE,MAAM,CAAC;IACf;;;sDAGkD;IAClD,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,iEAAiE;IACjE,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB;;;;OAIG;IACH,MAAM,CAAC,EACH,OAAO,0BAA0B,EAAE,WAAW,GAC9C,MAAM,GACN;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;KAAE,CAAC;IACtD,qDAAqD;IACrD,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,+DAA+D;IAC/D,gBAAgB,CAAC,EACb,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,GAC/B,CAAC,MACG,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,GAC/B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC,CAAC,CAAC;IAClD,kFAAkF;IAClF,KAAK,CAAC,EAAE,MAAM,CAAC;IACf;;;;;;;;;OASG;IACH,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,KAAK,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IACtE;;;;;;OAMG;IACH,cAAc,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,KAAK,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IACxE;;;;OAIG;IACH,iBAAiB,CAAC,EAAE,OAAO,CAAC;IAC5B;;;;;;;;;;;;;;OAcG;IACH,YAAY,CAAC,EAAE,CACb,KAAK,EAAE,GAAG,EACV,KAAK,EAAE,MAAM,KACV,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IAC5C;;;;OAIG;IACH,kBAAkB,CAAC,EAAE,OAAO,8BAA8B,EAAE,2BAA2B,CAAC;IACxF;;;;OAIG;IACH,cAAc,CAAC,EAAE,CAAC,OAAO,EAAE;QACzB,KAAK,EAAE,GAAG,CAAC;QACX,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;QAC1B,OAAO,EAAE,MAAM,CAAC;QAChB,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,WAAW,EAAE,mBAAmB,EAAE,CAAC;QACnC,UAAU,EAAE,kBAAkB,EAAE,CAAC;QACjC,QAAQ,CAAC,EAAE,MAAM,CAAC;QAClB,oBAAoB,CAAC,EAAE,OAAO,CAAC;QAC/B,IAAI,EAAE,KAAK,GAAG,MAAM,CAAC;KACtB,KACG,IAAI,GACJ;QACE,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,WAAW,CAAC,EAAE,mBAAmB,EAAE,CAAC;KACrC,GACD,OAAO,CAAC,IAAI,GAAG;QACb,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,WAAW,CAAC,EAAE,mBAAmB,EAAE,CAAC;KACrC,CAAC,CAAC;IACP;;;;;;;;;;;;;;OAcG;IACH,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB;;;;;;;;;;;;;OAaG;IACH,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB;;;;;;;;;;;;;;;;;;OAkBG;IACH,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAC7B;;;;;OAKG;IACH,kBAAkB,CAAC,EAAE,CAAC,OAAO,EAAE;QAC7B,OAAO,EAAE,OAAO,iBAAiB,EAAE,OAAO,CAAC;QAC3C,IAAI,EAAE,MAAM,CAAC;QACb,OAAO,EAAE,OAAO,iBAAiB,EAAE,iBAAiB,CAAC;QACrD,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;KAC/B,KACG,OAAO,iBAAiB,EAAE,OAAO,GACjC,MAAM,GACN,IAAI,GACJ,SAAS,GACT,OAAO,CAAC,OAAO,iBAAiB,EAAE,OAAO,GAAG,MAAM,GAAG,IAAI,GAAG,SAAS,CAAC,CAAC;CAC5E;AA0iBD;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,wBAAsB,sBAAsB,CAC1C,KAAK,EAAE,MAAM,EACb,OAAO,UAAQ,EACf,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,CAAC,CAoJjB;AAiPD,wBAAgB,sCAAsC,CAAC,KAAK,EAAE;IAC5D,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,IAAI,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CACtB,GAAG,OAAO,CA2BV;AAED,wBAAgB,qBAAqB,CACnC,OAAO,CAAC,EAAE,sBAAsB,GAC/B,cAAc,CA8vGhB;AAED;;;;GAIG;AACH,eAAO,MAAM,sBAAsB,EAAE,cAAwC,CAAC;AAa9E,yEAAyE;AACzE,wBAAgB,mBAAmB,IAAI,gBAAgB,GAAG,IAAI,CAE7D"}
1
+ {"version":3,"file":"agent-chat-plugin.d.ts","sourceRoot":"","sources":["../../src/server/agent-chat-plugin.ts"],"names":[],"mappings":"AAaA,OAAO,EASL,KAAK,WAAW,EACjB,MAAM,8BAA8B,CAAC;AAqBtC,OAAO,KAAK,EACV,mBAAmB,EACnB,cAAc,EACd,kBAAkB,EAElB,eAAe,EAEhB,MAAM,mBAAmB,CAAC;AAG3B,OAAO,EACL,gBAAgB,EAajB,MAAM,wBAAwB,CAAC;AA+DhC,OAAO,EAGL,KAAK,0BAA0B,EAC/B,KAAK,oBAAoB,EAC1B,MAAM,6BAA6B,CAAC;AA+SrC,wBAAgB,yBAAyB,CACvC,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GACnC,KAAK,CAAC;IACP,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,WAAW,CAAC,aAAa,CAAC,CAAC;CACzC,CAAC,CASD;AAmBD,wBAAgB,wBAAwB,CACtC,MAAM,EAAE,SAAS,cAAc,EAAE,EACjC,WAAW,EAAE,SAAS,oBAAoB,EAAE,EAC5C,OAAO,GAAE,0BAA0B,GAAG;IAAE,KAAK,CAAC,EAAE,GAAG,CAAA;CAAO,GACzD;IAAE,YAAY,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CAO7C;AA21CD,KAAK,cAAc,GAAG,CAAC,QAAQ,EAAE,GAAG,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;AAE9D,MAAM,WAAW,sBAAsB;IACrC,+DAA+D;IAC/D,OAAO,CAAC,EACJ,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,CAAC,MACG,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC;IAC9C,wCAAwC;IACxC,OAAO,CAAC,EACJ,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,CAAC,MACG,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,GAC3B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC;IAC9C,mEAAmE;IACnE,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,qDAAqD;IACrD,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,qEAAqE;IACrE,KAAK,CAAC,EAAE,MAAM,CAAC;IACf;;;sDAGkD;IAClD,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,iEAAiE;IACjE,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB;;;;OAIG;IACH,MAAM,CAAC,EACH,OAAO,0BAA0B,EAAE,WAAW,GAC9C,MAAM,GACN;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;KAAE,CAAC;IACtD,qDAAqD;IACrD,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,+DAA+D;IAC/D,gBAAgB,CAAC,EACb,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,GAC/B,CAAC,MACG,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,GAC/B,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC,CAAC,CAAC;IAClD,kFAAkF;IAClF,KAAK,CAAC,EAAE,MAAM,CAAC;IACf;;;;;;;;;OASG;IACH,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,KAAK,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IACtE;;;;;;OAMG;IACH,cAAc,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,KAAK,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IACxE;;;;OAIG;IACH,iBAAiB,CAAC,EAAE,OAAO,CAAC;IAC5B;;;;;;;;;;;;;;OAcG;IACH,YAAY,CAAC,EAAE,CACb,KAAK,EAAE,GAAG,EACV,KAAK,EAAE,MAAM,KACV,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;IAC5C;;;;OAIG;IACH,kBAAkB,CAAC,EAAE,OAAO,8BAA8B,EAAE,2BAA2B,CAAC;IACxF;;;;OAIG;IACH,cAAc,CAAC,EAAE,CAAC,OAAO,EAAE;QACzB,KAAK,EAAE,GAAG,CAAC;QACX,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;QAC1B,OAAO,EAAE,MAAM,CAAC;QAChB,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,WAAW,EAAE,mBAAmB,EAAE,CAAC;QACnC,UAAU,EAAE,kBAAkB,EAAE,CAAC;QACjC,QAAQ,CAAC,EAAE,MAAM,CAAC;QAClB,oBAAoB,CAAC,EAAE,OAAO,CAAC;QAC/B,IAAI,EAAE,KAAK,GAAG,MAAM,CAAC;KACtB,KACG,IAAI,GACJ;QACE,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,WAAW,CAAC,EAAE,mBAAmB,EAAE,CAAC;KACrC,GACD,OAAO,CAAC,IAAI,GAAG;QACb,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,WAAW,CAAC,EAAE,mBAAmB,EAAE,CAAC;KACrC,CAAC,CAAC;IACP;;;;;;;;;;;;;;OAcG;IACH,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB;;;;;;;;;;;;;OAaG;IACH,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB;;;;;;;;;;;;;;;;;;OAkBG;IACH,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAC7B;;;;;OAKG;IACH,kBAAkB,CAAC,EAAE,CAAC,OAAO,EAAE;QAC7B,OAAO,EAAE,OAAO,iBAAiB,EAAE,OAAO,CAAC;QAC3C,IAAI,EAAE,MAAM,CAAC;QACb,OAAO,EAAE,OAAO,iBAAiB,EAAE,iBAAiB,CAAC;QACrD,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;KAC/B,KACG,OAAO,iBAAiB,EAAE,OAAO,GACjC,MAAM,GACN,IAAI,GACJ,SAAS,GACT,OAAO,CAAC,OAAO,iBAAiB,EAAE,OAAO,GAAG,MAAM,GAAG,IAAI,GAAG,SAAS,CAAC,CAAC;CAC5E;AA4iBD;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,wBAAsB,sBAAsB,CAC1C,KAAK,EAAE,MAAM,EACb,OAAO,UAAQ,EACf,SAAS,CAAC,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,CAAC,CAoJjB;AAiPD,wBAAgB,sCAAsC,CAAC,KAAK,EAAE;IAC5D,OAAO,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,IAAI,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CACtB,GAAG,OAAO,CA2BV;AAED,wBAAgB,qBAAqB,CACnC,OAAO,CAAC,EAAE,sBAAsB,GAC/B,cAAc,CAs0GhB;AAED;;;;GAIG;AACH,eAAO,MAAM,sBAAsB,EAAE,cAAwC,CAAC;AAa9E,yEAAyE;AACzE,wBAAgB,mBAAmB,IAAI,gBAAgB,GAAG,IAAI,CAE7D"}
@@ -14,7 +14,7 @@ import { buildAssistantMessage, buildUserMessage, extractThreadMeta, mergeThread
14
14
  import { createError, defineEventHandler, setResponseStatus, setResponseHeader, getMethod, getQuery, getHeader, } from "h3";
15
15
  import { getSession } from "./auth.js";
16
16
  import { getOrigin } from "./google-oauth.js";
17
- import { createThread, forkThread, getThread, listThreads, searchThreads, setThreadScope, updateThreadData, withThreadDataLock, deleteThread, setThreadQueuedMessages, } from "../chat-threads/store.js";
17
+ import { createThread, forkThread, getThread, listThreads, searchThreads, renameThread, setThreadArchived, setThreadPinned, setThreadScope, updateThreadData, withThreadDataLock, deleteThread, setThreadQueuedMessages, } from "../chat-threads/store.js";
18
18
  import { resourceList, resourceListAccessible, resourceGet, resourceGetByPath, ensurePersonalDefaults, SHARED_OWNER, WORKSPACE_OWNER, } from "../resources/store.js";
19
19
  import { getFrontmatterValue, getSkillNameFromPath, parseFrontmatter, } from "../resources/metadata.js";
20
20
  import nodePath from "node:path";
@@ -880,7 +880,8 @@ async function createResourceScriptEntries() {
880
880
  }
881
881
  }
882
882
  /**
883
- * Creates a unified chat-history ActionEntry that dispatches to search or open.
883
+ * Creates a unified chat-history ActionEntry that dispatches to search, open,
884
+ * rename, or lightweight organization actions.
884
885
  */
885
886
  async function createChatScriptEntries() {
886
887
  try {
@@ -925,14 +926,14 @@ async function createChatScriptEntries() {
925
926
  return {
926
927
  "chat-history": {
927
928
  tool: {
928
- description: "Manage past agent chat threads. Use action 'search' to find previous conversations by keyword, or 'open' to open a thread in the UI.",
929
+ description: "Manage past agent chat threads. Use action 'search' to find previous conversations by keyword, 'open' to open a thread in the UI, or 'rename'/'pin'/'unpin'/'archive' to organize history.",
929
930
  parameters: {
930
931
  type: "object",
931
932
  properties: {
932
933
  action: {
933
934
  type: "string",
934
935
  description: "The operation to perform",
935
- enum: ["search", "open"],
936
+ enum: ["search", "open", "rename", "pin", "unpin", "archive"],
936
937
  },
937
938
  query: {
938
939
  type: "string",
@@ -949,7 +950,11 @@ async function createChatScriptEntries() {
949
950
  },
950
951
  id: {
951
952
  type: "string",
952
- description: "(open) The chat thread ID to open",
953
+ description: "(open, rename, pin, unpin, archive) The chat thread ID to manage",
954
+ },
955
+ title: {
956
+ type: "string",
957
+ description: "(rename) New chat title",
953
958
  },
954
959
  },
955
960
  required: ["action"],
@@ -959,6 +964,41 @@ async function createChatScriptEntries() {
959
964
  if (args?.action === "open") {
960
965
  return openEntry.run(args);
961
966
  }
967
+ if (args?.action === "rename" ||
968
+ args?.action === "pin" ||
969
+ args?.action === "unpin" ||
970
+ args?.action === "archive") {
971
+ const id = typeof args?.id === "string" ? args.id : "";
972
+ if (!id)
973
+ return "Missing required id.";
974
+ const owner = getRequestRunContext()?.owner ?? getRequestUserEmail() ?? "";
975
+ if (!owner)
976
+ return "No authenticated user is available.";
977
+ const thread = await getThread(id);
978
+ if (!thread || thread.ownerEmail !== owner) {
979
+ return `Chat thread "${id}" not found.`;
980
+ }
981
+ const title = thread.title || thread.preview || "(untitled)";
982
+ if (args.action === "rename") {
983
+ const nextTitle = typeof args?.title === "string"
984
+ ? args.title.replace(/\s+/g, " ").trim().slice(0, 160)
985
+ : "";
986
+ if (!nextTitle)
987
+ return "Missing required title.";
988
+ const renamed = await renameThread(id, nextTitle, {
989
+ ownerEmail: owner,
990
+ });
991
+ if (!renamed)
992
+ return `Chat thread "${id}" could not be renamed.`;
993
+ return `Renamed chat "${title}" to "${nextTitle}".`;
994
+ }
995
+ if (args.action === "archive") {
996
+ await setThreadArchived(id, true);
997
+ return `Archived chat: ${title}`;
998
+ }
999
+ await setThreadPinned(id, args.action === "pin");
1000
+ return `${args.action === "pin" ? "Pinned" : "Unpinned"} chat: ${title}`;
1001
+ }
962
1002
  return searchEntry.run(args);
963
1003
  },
964
1004
  },
@@ -1488,6 +1528,7 @@ Use for charts, visualizations, previews. Don't use for simple text/tables or ex
1488
1528
  You can search and restore previous chat conversations using \`chat-history\`:
1489
1529
  - \`chat-history\` (action: "search") — Search or list past chat threads by keyword
1490
1530
  - \`chat-history\` (action: "open") — Open a chat thread in the UI as a new tab and focus it
1531
+ - \`chat-history\` (actions: "rename", "pin", "unpin", "archive") — Organize a known chat thread by ID
1491
1532
 
1492
1533
  When the user asks to find a previous conversation, use \`chat-history\` with action "search" first to find matching threads, then action "open" to restore the one they want.`,
1493
1534
  "agent-teams": `### Agent Teams — Orchestration
@@ -1680,6 +1721,7 @@ Which routes are renderable as embeds is template-specific — the app's \`AGENT
1680
1721
  You can search and restore previous chat conversations using \`chat-history\`:
1681
1722
  - \`chat-history\` (action: "search") — Search or list past chat threads by keyword
1682
1723
  - \`chat-history\` (action: "open") — Open a chat thread in the UI as a new tab and focus it
1724
+ - \`chat-history\` (actions: "rename", "pin", "unpin", "archive") — Organize a known chat thread by ID
1683
1725
 
1684
1726
  When the user asks to find a previous conversation, use \`chat-history\` with action "search" first to find matching threads, then action "open" to restore the one they want.
1685
1727
 
@@ -4766,6 +4808,21 @@ Non-code requests are still fine on this surface: read data, navigate the UI, su
4766
4808
  const body = await readBody(event);
4767
4809
  let newThreadData = body.threadData || thread.threadData;
4768
4810
  let newMessageCount = body.messageCount ?? thread.messageCount;
4811
+ let nextTitle = typeof body.title === "string" ? body.title : thread.title;
4812
+ const nextPreview = typeof body.preview === "string"
4813
+ ? body.preview
4814
+ : thread.preview;
4815
+ const preserveTitleOverride = (repo) => {
4816
+ if (repo &&
4817
+ typeof repo === "object" &&
4818
+ typeof repo
4819
+ ._titleOverride === "string" &&
4820
+ repo._titleOverride.trim()) {
4821
+ const meta = extractThreadMeta(repo);
4822
+ if (meta.title)
4823
+ nextTitle = meta.title;
4824
+ }
4825
+ };
4769
4826
  // Merge the incoming full-thread blob over the current SQL
4770
4827
  // copy. Periodic saves can be stale relative to server-side
4771
4828
  // run completion, and threadRuntime.export() does not carry
@@ -4779,12 +4836,21 @@ Non-code requests are still fine on this surface: read data, navigate the UI, su
4779
4836
  if (Array.isArray(merged.messages)) {
4780
4837
  newMessageCount = merged.messages.length;
4781
4838
  }
4839
+ preserveTitleOverride(merged);
4782
4840
  }
4783
4841
  catch {
4784
4842
  // Invalid JSON in either side — fall back to raw body blob.
4785
4843
  }
4786
4844
  }
4787
- await updateThreadData(threadId, newThreadData, body.title ?? thread.title, body.preview ?? thread.preview, newMessageCount);
4845
+ else {
4846
+ try {
4847
+ preserveTitleOverride(JSON.parse(newThreadData));
4848
+ }
4849
+ catch {
4850
+ // Invalid JSON — keep the title supplied by the client.
4851
+ }
4852
+ }
4853
+ await updateThreadData(threadId, newThreadData, nextTitle, nextPreview, newMessageCount);
4788
4854
  // Scope updates piggyback on the PUT — the client uses this
4789
4855
  // path for both "detach" (scope: null) and "retag" flows.
4790
4856
  // Send the field as `scope: undefined` (or omit it) when
@@ -4813,6 +4879,49 @@ Non-code requests are still fine on this surface: read data, navigate the UI, su
4813
4879
  await setThreadQueuedMessages(threadId, queued);
4814
4880
  return { ok: true };
4815
4881
  }
4882
+ if (method === "POST" && isThreadSubroute("rename")) {
4883
+ const thread = await getThread(threadId);
4884
+ if (!thread || thread.ownerEmail !== owner) {
4885
+ setResponseStatus(event, 404);
4886
+ return { error: "Thread not found" };
4887
+ }
4888
+ const body = await readBody(event).catch(() => ({}));
4889
+ const title = typeof body?.title === "string"
4890
+ ? body.title.replace(/\s+/g, " ").trim().slice(0, 160)
4891
+ : "";
4892
+ if (!title) {
4893
+ setResponseStatus(event, 400);
4894
+ return { error: "Title is required" };
4895
+ }
4896
+ const renamed = await renameThread(threadId, title, {
4897
+ ownerEmail: owner,
4898
+ });
4899
+ if (!renamed) {
4900
+ setResponseStatus(event, 404);
4901
+ return { error: "Thread not found" };
4902
+ }
4903
+ return { ok: true };
4904
+ }
4905
+ if (method === "POST" && isThreadSubroute("pin")) {
4906
+ const thread = await getThread(threadId);
4907
+ if (!thread || thread.ownerEmail !== owner) {
4908
+ setResponseStatus(event, 404);
4909
+ return { error: "Thread not found" };
4910
+ }
4911
+ const body = await readBody(event).catch(() => ({}));
4912
+ await setThreadPinned(threadId, body?.pinned !== false);
4913
+ return { ok: true };
4914
+ }
4915
+ if (method === "POST" && isThreadSubroute("archive")) {
4916
+ const thread = await getThread(threadId);
4917
+ if (!thread || thread.ownerEmail !== owner) {
4918
+ setResponseStatus(event, 404);
4919
+ return { error: "Thread not found" };
4920
+ }
4921
+ const body = await readBody(event).catch(() => ({}));
4922
+ await setThreadArchived(threadId, body?.archived !== false);
4923
+ return { ok: true };
4924
+ }
4816
4925
  // POST /threads/:id/fork — duplicate a thread with all its messages
4817
4926
  if (method === "POST" && isThreadSubroute("fork")) {
4818
4927
  const body = await readBody(event);