@assistant-ui/react-ai-sdk 0.1.0 → 0.1.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.
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/core/VercelModelAdapter.tsx","../src/rsc/useVercelRSCRuntime.tsx","../src/rsc/VercelRSCRuntime.tsx","../src/rsc/useVercelRSCSync.tsx","../src/utils/ThreadMessageConverter.ts","../src/rsc/getVercelRSCMessage.tsx","../src/ui/use-chat/useVercelUseChatRuntime.tsx","../src/ui/VercelAIRuntime.tsx","../src/ui/getVercelAIMessage.tsx","../src/ui/utils/sliceMessagesUntil.tsx","../src/ui/utils/useVercelAIComposerSync.tsx","../src/ui/utils/useVercelAIThreadSync.tsx","../src/ui/use-assistant/useVercelUseAssistantRuntime.tsx"],"sourcesContent":["\"use client\";\n\nimport type {\n AssistantContentPart,\n TextContentPart,\n} from \"@assistant-ui/react\";\nimport type {\n ChatModelAdapter,\n ChatModelRunOptions,\n} from \"@assistant-ui/react\";\nimport { type LanguageModel, streamText } from \"ai\";\n\nexport class VercelModelAdapter implements ChatModelAdapter {\n constructor(private readonly model: LanguageModel) {}\n\n async run({ messages, abortSignal, onUpdate }: ChatModelRunOptions) {\n const { fullStream } = await streamText({\n model: this.model,\n abortSignal,\n messages: messages.map((m) => ({\n role: m.role,\n content: m.content.filter((c): c is TextContentPart => c.type !== \"ui\"),\n })),\n });\n\n const content: AssistantContentPart[] = [];\n for await (const aiPart of fullStream) {\n switch (aiPart.type) {\n case \"text-delta\": {\n let part = content.at(-1);\n if (!part || part.type !== \"text\") {\n part = { type: \"text\", text: \"\" };\n content.push(part);\n }\n part.text += aiPart.textDelta;\n break;\n }\n // TODO tool results\n case \"tool-call\": {\n content.push({\n type: \"tool-call\",\n name: aiPart.toolName,\n args: aiPart.args,\n });\n break;\n }\n }\n\n onUpdate({ content });\n }\n\n return { content };\n }\n}\n","\"use client\";\n\nimport { useEffect, useInsertionEffect, useState } from \"react\";\nimport type { VercelRSCAdapter } from \"./VercelRSCAdapter\";\nimport { VercelRSCRuntime } from \"./VercelRSCRuntime\";\n\nexport const useVercelRSCRuntime = <T extends WeakKey>(\n adapter: VercelRSCAdapter<T>,\n) => {\n const [runtime] = useState(() => new VercelRSCRuntime(adapter));\n\n useInsertionEffect(() => {\n runtime.adapter = adapter;\n });\n useEffect(() => {\n runtime.onAdapterUpdated();\n });\n\n return runtime;\n};\n","\"use client\";\n\nimport type {\n AssistantRuntime,\n ReactThreadRuntime,\n Unsubscribe,\n} from \"@assistant-ui/react\";\n\nimport type { AppendMessage, ThreadMessage } from \"@assistant-ui/react\";\nimport { ProxyConfigProvider } from \"@assistant-ui/react/internal\";\nimport { type StoreApi, type UseBoundStore, create } from \"zustand\";\nimport type { VercelRSCAdapter } from \"./VercelRSCAdapter\";\nimport type { VercelRSCMessage } from \"./VercelRSCMessage\";\nimport { useVercelRSCSync } from \"./useVercelRSCSync\";\n\nconst EMPTY_BRANCHES: readonly never[] = Object.freeze([]);\n\nexport class VercelRSCRuntime<T extends WeakKey = VercelRSCMessage>\n extends ProxyConfigProvider\n implements AssistantRuntime, ReactThreadRuntime\n{\n private useAdapter: UseBoundStore<StoreApi<{ adapter: VercelRSCAdapter<T> }>>;\n\n private _subscriptions = new Set<() => void>();\n\n public isRunning = false;\n public messages: ThreadMessage[] = [];\n\n constructor(public adapter: VercelRSCAdapter<T>) {\n super();\n\n this.useAdapter = create(() => ({\n adapter,\n }));\n }\n\n private withRunning = (callback: Promise<unknown>) => {\n this.isRunning = true;\n return callback.finally(() => {\n this.isRunning = false;\n });\n };\n\n public getBranches(): readonly string[] {\n return EMPTY_BRANCHES;\n }\n\n public switchToBranch(): void {\n throw new Error(\n \"Branch switching is not supported by VercelRSCAssistantProvider.\",\n );\n }\n\n public async append(message: AppendMessage): Promise<void> {\n if (message.parentId !== (this.messages.at(-1)?.id ?? null)) {\n if (!this.adapter.edit)\n throw new Error(\n \"Message editing is not enabled, please provide an edit callback to VercelRSCAssistantProvider.\",\n );\n await this.withRunning(this.adapter.edit(message));\n } else {\n await this.withRunning(this.adapter.append(message));\n }\n }\n\n public async startRun(parentId: string | null): Promise<void> {\n if (!this.adapter.reload)\n throw new Error(\n \"Message reloading is not enabled, please provide a reload callback to VercelRSCAssistantProvider.\",\n );\n await this.withRunning(this.adapter.reload(parentId));\n }\n\n cancelRun(): void {\n // in dev mode, log a warning\n if (process.env[\"NODE_ENV\"] === \"development\") {\n console.warn(\n \"Run cancellation is not supported by VercelRSCAssistantProvider.\",\n );\n }\n }\n\n public subscribe(callback: () => void): Unsubscribe {\n this._subscriptions.add(callback);\n return () => this._subscriptions.delete(callback);\n }\n\n public onAdapterUpdated() {\n if (this.useAdapter.getState().adapter !== this.adapter) {\n this.useAdapter.setState({ adapter: this.adapter });\n }\n }\n\n private updateData = (messages: ThreadMessage[]) => {\n this.messages = messages;\n for (const callback of this._subscriptions) callback();\n };\n\n unstable_synchronizer = () => {\n const { adapter } = this.useAdapter();\n\n useVercelRSCSync(adapter, this.updateData);\n\n return null;\n };\n}\n","import type { ThreadMessage } from \"@assistant-ui/react\";\nimport { useEffect, useMemo } from \"react\";\nimport {\n type ConverterCallback,\n ThreadMessageConverter,\n} from \"../utils/ThreadMessageConverter\";\nimport type { VercelRSCAdapter } from \"./VercelRSCAdapter\";\nimport type { VercelRSCMessage } from \"./VercelRSCMessage\";\nimport {\n type VercelRSCThreadMessage,\n symbolInnerRSCMessage,\n} from \"./getVercelRSCMessage\";\n\nconst vercelToThreadMessage = <T,>(\n converter: (message: T) => VercelRSCMessage,\n rawMessage: T,\n): VercelRSCThreadMessage<T> => {\n const message = converter(rawMessage);\n\n return {\n id: message.id,\n role: message.role,\n content: [{ type: \"ui\", display: message.display }],\n createdAt: message.createdAt ?? new Date(),\n ...{ status: \"done\" },\n [symbolInnerRSCMessage]: rawMessage,\n };\n};\n\ntype UpdateDataCallback = (messages: ThreadMessage[]) => void;\n\nexport const useVercelRSCSync = <T extends WeakKey>(\n adapter: VercelRSCAdapter<T>,\n updateData: UpdateDataCallback,\n) => {\n // flush the converter cache when the convertMessage prop changes\n const [converter, convertCallback] = useMemo(() => {\n const rscConverter =\n adapter.convertMessage ?? ((m: T) => m as VercelRSCMessage);\n const convertCallback: ConverterCallback<T> = (m, cache) => {\n if (cache) return cache;\n return vercelToThreadMessage(rscConverter, m);\n };\n return [new ThreadMessageConverter(), convertCallback];\n }, [adapter.convertMessage]);\n\n useEffect(() => {\n updateData(converter.convertMessages(adapter.messages, convertCallback));\n }, [updateData, converter, convertCallback, adapter.messages]);\n};\n","import type { ThreadMessage } from \"@assistant-ui/react\";\n\nexport type ConverterCallback<TIn> = (\n message: TIn,\n cache: ThreadMessage | undefined,\n) => ThreadMessage;\n\nexport class ThreadMessageConverter {\n private readonly cache = new WeakMap<WeakKey, ThreadMessage>();\n\n convertMessages<TIn extends WeakKey>(\n messages: TIn[],\n converter: ConverterCallback<TIn>,\n keyMapper: (m: TIn) => WeakKey = (key) => key,\n ): ThreadMessage[] {\n return messages.map((m) => {\n const key = keyMapper(m);\n const cached = this.cache.get(key);\n const newMessage = converter(m, cached);\n this.cache.set(key, newMessage);\n return newMessage;\n });\n }\n}\n","import type { ThreadMessage } from \"@assistant-ui/react\";\n\nexport const symbolInnerRSCMessage = Symbol(\"innerVercelRSCMessage\");\n\nexport type VercelRSCThreadMessage<T> = ThreadMessage & {\n [symbolInnerRSCMessage]?: T;\n};\n\nexport const getVercelRSCMessage = <T,>(message: ThreadMessage) => {\n return (message as VercelRSCThreadMessage<T>)[symbolInnerRSCMessage];\n};\n","import type { UseChatHelpers } from \"@ai-sdk/react\";\nimport { useEffect, useInsertionEffect, useState } from \"react\";\nimport { VercelAIRuntime } from \"../VercelAIRuntime\";\n\nexport const useVercelUseChatRuntime = (chatHelpers: UseChatHelpers) => {\n const [runtime] = useState(() => new VercelAIRuntime(chatHelpers));\n\n useInsertionEffect(() => {\n runtime.vercel = chatHelpers;\n });\n useEffect(() => {\n runtime.onVercelUpdated();\n });\n\n return runtime;\n};\n","import type {\n AssistantRuntime,\n ReactThreadRuntime,\n Unsubscribe,\n} from \"@assistant-ui/react\";\nimport type { AppendMessage, ThreadMessage } from \"@assistant-ui/react\";\nimport {\n MessageRepository,\n ProxyConfigProvider,\n} from \"@assistant-ui/react/internal\";\nimport type { Message } from \"ai\";\nimport { type StoreApi, type UseBoundStore, create } from \"zustand\";\nimport { getVercelAIMessage } from \"./getVercelAIMessage\";\nimport type { VercelHelpers } from \"./utils/VercelHelpers\";\nimport { sliceMessagesUntil } from \"./utils/sliceMessagesUntil\";\nimport { useVercelAIComposerSync } from \"./utils/useVercelAIComposerSync\";\nimport { useVercelAIThreadSync } from \"./utils/useVercelAIThreadSync\";\n\nconst hasUpcomingMessage = (isRunning: boolean, messages: ThreadMessage[]) => {\n return isRunning && messages[messages.length - 1]?.role !== \"assistant\";\n};\n\nexport class VercelAIRuntime\n extends ProxyConfigProvider\n implements AssistantRuntime, ReactThreadRuntime\n{\n private _subscriptions = new Set<() => void>();\n private repository = new MessageRepository();\n private assistantOptimisticId: string | null = null;\n\n private useVercel: UseBoundStore<StoreApi<{ vercel: VercelHelpers }>>;\n\n public messages: ThreadMessage[] = [];\n public isRunning = false;\n\n constructor(public vercel: VercelHelpers) {\n super();\n this.useVercel = create(() => ({\n vercel,\n }));\n }\n\n public getBranches(messageId: string): string[] {\n return this.repository.getBranches(messageId);\n }\n\n public switchToBranch(branchId: string): void {\n this.repository.switchToBranch(branchId);\n this.updateVercelMessages(this.repository.getMessages());\n }\n\n public async append(message: AppendMessage): Promise<void> {\n // add user message\n if (message.content.length !== 1 || message.content[0]?.type !== \"text\")\n throw new Error(\"Only text content is supported by Vercel AI SDK.\");\n\n const newMessages = sliceMessagesUntil(\n this.vercel.messages,\n message.parentId,\n );\n this.vercel.setMessages(newMessages);\n\n await this.vercel.append({\n role: \"user\",\n content: message.content[0].text,\n });\n }\n\n public async startRun(parentId: string | null): Promise<void> {\n const reloadMaybe =\n \"reload\" in this.vercel ? this.vercel.reload : undefined;\n if (!reloadMaybe)\n throw new Error(\n \"Reload is not supported by Vercel AI SDK's useAssistant.\",\n );\n\n const newMessages = sliceMessagesUntil(this.vercel.messages, parentId);\n this.vercel.setMessages(newMessages);\n\n await reloadMaybe();\n }\n\n public cancelRun(): void {\n const previousMessage = this.vercel.messages.at(-1);\n\n this.vercel.stop();\n\n if (this.assistantOptimisticId) {\n this.repository.deleteMessage(this.assistantOptimisticId);\n this.assistantOptimisticId = null;\n }\n\n let messages = this.repository.getMessages();\n if (\n previousMessage?.role === \"user\" &&\n previousMessage.id === messages.at(-1)?.id // ensure the previous message is a leaf node\n ) {\n this.vercel.setInput(previousMessage.content);\n this.repository.deleteMessage(previousMessage.id);\n\n messages = this.repository.getMessages();\n }\n\n // resync messages\n setTimeout(() => {\n this.updateVercelMessages(messages);\n }, 0);\n }\n\n public subscribe(callback: () => void): Unsubscribe {\n this._subscriptions.add(callback);\n return () => this._subscriptions.delete(callback);\n }\n\n private updateVercelMessages = (messages: ThreadMessage[]) => {\n this.vercel.setMessages(\n messages\n .flatMap(getVercelAIMessage)\n .filter((m): m is Message => m != null),\n );\n };\n\n public onVercelUpdated() {\n if (this.useVercel.getState().vercel !== this.vercel) {\n this.useVercel.setState({ vercel: this.vercel });\n }\n }\n\n private updateData = (isRunning: boolean, vm: ThreadMessage[]) => {\n for (let i = 0; i < vm.length; i++) {\n const message = vm[i]!;\n const parent = vm[i - 1];\n this.repository.addOrUpdateMessage(parent?.id ?? null, message);\n }\n\n if (this.assistantOptimisticId) {\n this.repository.deleteMessage(this.assistantOptimisticId);\n this.assistantOptimisticId = null;\n }\n\n if (hasUpcomingMessage(isRunning, vm)) {\n this.assistantOptimisticId = this.repository.appendOptimisticMessage(\n vm.at(-1)?.id ?? null,\n {\n role: \"assistant\",\n content: [{ type: \"text\", text: \"\" }],\n },\n );\n }\n\n this.repository.resetHead(\n this.assistantOptimisticId ?? vm.at(-1)?.id ?? null,\n );\n\n this.messages = this.repository.getMessages();\n this.isRunning = isRunning;\n\n for (const callback of this._subscriptions) callback();\n };\n\n unstable_synchronizer = () => {\n const { vercel } = this.useVercel();\n\n useVercelAIThreadSync(vercel, this.updateData);\n useVercelAIComposerSync(vercel);\n\n return null;\n };\n}\n","import type { ThreadMessage } from \"@assistant-ui/react\";\nimport type { Message } from \"ai\";\n\nexport const symbolInnerAIMessage = Symbol(\"innerVercelAIUIMessage\");\n\nexport type VercelAIThreadMessage = ThreadMessage & {\n [symbolInnerAIMessage]?: Message[];\n};\n\nexport const getVercelAIMessage = (message: ThreadMessage) => {\n return (message as VercelAIThreadMessage)[symbolInnerAIMessage];\n};\n","import type { Message } from \"ai\";\n\nexport const sliceMessagesUntil = (\n messages: Message[],\n messageId: string | null,\n) => {\n if (messageId == null) return [];\n\n let messageIdx = messages.findIndex((m) => m.id === messageId);\n if (messageIdx === -1)\n throw new Error(\n \"useVercelAIThreadState: Message not found. This is liekly an internal bug in assistant-ui.\",\n );\n\n while (messages[messageIdx + 1]?.role === \"assistant\") {\n messageIdx++;\n }\n\n return messages.slice(0, messageIdx + 1);\n};\n","import { useThreadContext } from \"@assistant-ui/react/experimental\";\nimport { useEffect } from \"react\";\nimport type { VercelHelpers } from \"./VercelHelpers\";\n\n// two way sync between vercel helpers input state and composer text state\nexport const useVercelAIComposerSync = (vercel: VercelHelpers) => {\n const { useComposer } = useThreadContext();\n\n useEffect(() => {\n useComposer.setState({\n value: vercel.input,\n setValue: vercel.setInput,\n });\n }, [useComposer, vercel.input, vercel.setInput]);\n};\n","import type { TextContentPart, ThreadMessage } from \"@assistant-ui/react\";\nimport type { ToolCallContentPart } from \"@assistant-ui/react/experimental\";\nimport type { Message } from \"ai\";\nimport { useEffect, useMemo } from \"react\";\nimport {\n type ConverterCallback,\n ThreadMessageConverter,\n} from \"../../utils/ThreadMessageConverter\";\nimport {\n type VercelAIThreadMessage,\n symbolInnerAIMessage,\n} from \"../getVercelAIMessage\";\nimport type { VercelHelpers } from \"./VercelHelpers\";\n\nconst getIsRunning = (vercel: VercelHelpers) => {\n if (\"isLoading\" in vercel) return vercel.isLoading;\n return vercel.status === \"in_progress\";\n};\n\nconst vercelToThreadMessage = (\n messages: Message[],\n status: \"in_progress\" | \"done\" | \"error\",\n): VercelAIThreadMessage => {\n const firstMessage = messages[0];\n if (!firstMessage) throw new Error(\"No messages found\");\n\n const common = {\n id: firstMessage.id,\n createdAt: firstMessage.createdAt ?? new Date(),\n [symbolInnerAIMessage]: messages,\n };\n\n switch (firstMessage.role) {\n case \"user\":\n if (messages.length > 1) {\n throw new Error(\n \"Multiple user messages found. This is likely an internal bug in assistant-ui.\",\n );\n }\n\n return {\n ...common,\n role: \"user\",\n content: [{ type: \"text\", text: firstMessage.content }],\n };\n case \"assistant\":\n return {\n ...common,\n role: \"assistant\",\n content: messages.flatMap((message) => [\n ...(message.content\n ? [{ type: \"text\", text: message.content } as TextContentPart]\n : []),\n ...(message.toolInvocations?.map(\n (t) =>\n ({\n type: \"tool-call\",\n name: t.toolName,\n args: t.args,\n result: \"result\" in t ? t.result : undefined,\n }) as ToolCallContentPart,\n ) ?? []),\n ]),\n status,\n };\n default:\n throw new Error(\n `You have a message with an unsupported role. The role ${firstMessage.role} is not supported.`,\n );\n }\n};\n\ntype Chunk = [Message, ...Message[]];\nconst hasItems = (messages: Message[]): messages is Chunk =>\n messages.length > 0;\n\nconst chunkedMessages = (messages: Message[]): Chunk[] => {\n const chunks: Chunk[] = [];\n let currentChunk: Message[] = [];\n\n for (const message of messages) {\n if (message.role === \"assistant\") {\n currentChunk.push(message);\n } else {\n if (hasItems(currentChunk)) {\n chunks.push(currentChunk);\n currentChunk = [];\n }\n chunks.push([message]);\n }\n }\n\n if (hasItems(currentChunk)) {\n chunks.push(currentChunk);\n }\n\n return chunks;\n};\n\nconst shallowArrayEqual = (a: unknown[], b: unknown[]) => {\n if (a.length !== b.length) return false;\n for (let i = 0; i < a.length; i++) {\n if (a[i] !== b[i]) return false;\n }\n return true;\n};\n\ntype UpdateDataCallback = (isRunning: boolean, vm: ThreadMessage[]) => void;\n\nexport const useVercelAIThreadSync = (\n vercel: VercelHelpers,\n updateData: UpdateDataCallback,\n) => {\n const isRunning = getIsRunning(vercel);\n\n const converter = useMemo(() => new ThreadMessageConverter(), []);\n\n useEffect(() => {\n const lastMessageId = vercel.messages.at(-1)?.id;\n const convertCallback: ConverterCallback<Chunk> = (messages, cache) => {\n const status =\n lastMessageId === messages[0].id && isRunning ? \"in_progress\" : \"done\";\n\n if (\n cache &&\n shallowArrayEqual(cache.content, messages) &&\n (cache.role === \"user\" || cache.status === status)\n )\n return cache;\n\n return vercelToThreadMessage(messages, status);\n };\n\n const messages = converter.convertMessages(\n chunkedMessages(vercel.messages),\n convertCallback,\n (m) => m[0],\n );\n\n updateData(isRunning, messages);\n }, [updateData, isRunning, vercel.messages, converter]);\n};\n","import type { UseAssistantHelpers } from \"@ai-sdk/react\";\nimport { useEffect, useInsertionEffect, useState } from \"react\";\nimport { VercelAIRuntime } from \"../VercelAIRuntime\";\n\nexport const useVercelUseAssistantRuntime = (\n assistantHelpers: UseAssistantHelpers,\n) => {\n const [runtime] = useState(() => new VercelAIRuntime(assistantHelpers));\n\n useInsertionEffect(() => {\n runtime.vercel = assistantHelpers;\n });\n useEffect(() => {\n runtime.onVercelUpdated();\n });\n\n return runtime;\n};\n"],"mappings":"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","names":["useEffect","convertCallback","useEffect","useEffect","useInsertionEffect","useState","ProxyConfigProvider","create","useEffect","useEffect","useMemo","vercelToThreadMessage","useMemo","useEffect","messages","ProxyConfigProvider","create","useState","useInsertionEffect","useEffect","useEffect","useInsertionEffect","useState","useState","useInsertionEffect","useEffect"]}
package/package.json CHANGED
@@ -1,8 +1,33 @@
1
1
  {
2
2
  "name": "@assistant-ui/react-ai-sdk",
3
- "version": "0.1.0",
3
+ "version": "0.1.2",
4
4
  "license": "MIT",
5
- "type": "module",
5
+ "exports": {
6
+ ".": {
7
+ "import": {
8
+ "types": "./dist/index.d.mts",
9
+ "default": "./dist/index.mjs"
10
+ },
11
+ "require": {
12
+ "types": "./dist/index.d.ts",
13
+ "default": "./dist/index.js"
14
+ }
15
+ }
16
+ },
17
+ "source": "./src/index.ts",
18
+ "main": "./dist/index.js",
19
+ "module": "./dist/index.mjs",
20
+ "types": "./dist/index.d.ts",
21
+ "files": [
22
+ "dist",
23
+ "README.md"
24
+ ],
25
+ "sideEffects": false,
26
+ "dependencies": {
27
+ "tsup": "^8.1.0",
28
+ "zod": "^3.23.8",
29
+ "zustand": "^4.5.2"
30
+ },
6
31
  "peerDependencies": {
7
32
  "@ai-sdk/react": "^0.0.x",
8
33
  "@assistant-ui/react": "^0.1",
@@ -15,19 +40,22 @@
15
40
  "optional": true
16
41
  }
17
42
  },
18
- "dependencies": {
19
- "tsup": "^8.1.0",
20
- "zod": "^3.23.8",
21
- "zustand": "^4.5.2"
22
- },
23
43
  "devDependencies": {
24
44
  "@types/node": "^20.14.2",
25
- "@assistant-ui/react": "0.1.0",
45
+ "@assistant-ui/react": "0.1.1",
26
46
  "@assistant-ui/tsconfig": "0.0.0"
27
47
  },
28
48
  "publishConfig": {
29
49
  "access": "public"
30
50
  },
51
+ "homepage": "https://assistant-ui.com/",
52
+ "repository": {
53
+ "type": "git",
54
+ "url": "git+https://github.com/Yonom/assistant-ui.git"
55
+ },
56
+ "bugs": {
57
+ "url": "https://github.com/Yonom/assistant-ui/issues"
58
+ },
31
59
  "scripts": {
32
60
  "build": "tsup src/index.ts --format cjs,esm --dts --sourcemap"
33
61
  }
@@ -1,20 +0,0 @@
1
-
2
- > @assistant-ui/react-ai-sdk@0.1.0 build /Users/yonom/Documents/GitHub/assistant-ui/packages/react-ai-sdk
3
- > tsup src/index.ts --format cjs,esm --dts --sourcemap
4
-
5
- CLI Building entry: src/index.ts
6
- CLI Using tsconfig: tsconfig.json
7
- CLI tsup v8.1.0
8
- CLI Target: esnext
9
- CJS Build start
10
- ESM Build start
11
- ESM dist/index.js 14.09 KB
12
- ESM dist/index.js.map 29.53 KB
13
- ESM ⚡️ Build success in 19ms
14
- CJS dist/index.cjs 15.44 KB
15
- CJS dist/index.cjs.map 29.32 KB
16
- CJS ⚡️ Build success in 19ms
17
- DTS Build start
18
- DTS ⚡️ Build success in 2227ms
19
- DTS dist/index.d.cts 3.45 KB
20
- DTS dist/index.d.ts 3.45 KB
@@ -1,54 +0,0 @@
1
- "use client";
2
-
3
- import type {
4
- AssistantContentPart,
5
- TextContentPart,
6
- } from "@assistant-ui/react";
7
- import type {
8
- ChatModelAdapter,
9
- ChatModelRunOptions,
10
- } from "@assistant-ui/react/experimental";
11
- import { type LanguageModel, streamText } from "ai";
12
-
13
- export class VercelModelAdapter implements ChatModelAdapter {
14
- constructor(private readonly model: LanguageModel) {}
15
-
16
- async run({ messages, abortSignal, onUpdate }: ChatModelRunOptions) {
17
- const { fullStream } = await streamText({
18
- model: this.model,
19
- abortSignal,
20
- messages: messages.map((m) => ({
21
- role: m.role,
22
- content: m.content.filter((c): c is TextContentPart => c.type !== "ui"),
23
- })),
24
- });
25
-
26
- const content: AssistantContentPart[] = [];
27
- for await (const aiPart of fullStream) {
28
- switch (aiPart.type) {
29
- case "text-delta": {
30
- let part = content.at(-1);
31
- if (!part || part.type !== "text") {
32
- part = { type: "text", text: "" };
33
- content.push(part);
34
- }
35
- part.text += aiPart.textDelta;
36
- break;
37
- }
38
- // TODO tool results
39
- case "tool-call": {
40
- content.push({
41
- type: "tool-call",
42
- name: aiPart.toolName,
43
- args: aiPart.args,
44
- });
45
- break;
46
- }
47
- }
48
-
49
- onUpdate({ content });
50
- }
51
-
52
- return { content };
53
- }
54
- }
package/src/core/index.ts DELETED
@@ -1 +0,0 @@
1
- export { VercelModelAdapter as unstable_VercelModelAdapter } from "./VercelModelAdapter";
package/src/index.ts DELETED
@@ -1,3 +0,0 @@
1
- export * from "./core";
2
- export * from "./rsc";
3
- export * from "./ui";
@@ -1,18 +0,0 @@
1
- "use client";
2
- import type { AppendMessage } from "@assistant-ui/react";
3
- import type { VercelRSCMessage } from "./VercelRSCMessage";
4
-
5
- type RSCMessageConverter<T> = {
6
- convertMessage: (message: T) => VercelRSCMessage;
7
- };
8
-
9
- type VercelRSCAdapterBase<T> = {
10
- messages: T[];
11
- append: (message: AppendMessage) => Promise<void>;
12
- edit?: (message: AppendMessage) => Promise<void>;
13
- reload?: (parentId: string | null) => Promise<void>;
14
- convertMessage?: (message: T) => VercelRSCMessage;
15
- };
16
-
17
- export type VercelRSCAdapter<T = VercelRSCMessage> = VercelRSCAdapterBase<T> &
18
- (T extends VercelRSCMessage ? object : RSCMessageConverter<T>);
@@ -1,9 +0,0 @@
1
- "use client";
2
- import type { ReactNode } from "react";
3
-
4
- export type VercelRSCMessage = {
5
- id: string;
6
- role: "user" | "assistant";
7
- display: ReactNode;
8
- createdAt?: Date;
9
- };
@@ -1,106 +0,0 @@
1
- "use client";
2
-
3
- import type {
4
- AssistantRuntime,
5
- ReactThreadRuntime,
6
- Unsubscribe,
7
- } from "@assistant-ui/react";
8
-
9
- import type { AppendMessage, ThreadMessage } from "@assistant-ui/react";
10
- import { ProxyConfigProvider } from "@assistant-ui/react/internal";
11
- import { type StoreApi, type UseBoundStore, create } from "zustand";
12
- import type { VercelRSCAdapter } from "./VercelRSCAdapter";
13
- import type { VercelRSCMessage } from "./VercelRSCMessage";
14
- import { useVercelRSCSync } from "./useVercelRSCSync";
15
-
16
- const EMPTY_BRANCHES: readonly never[] = Object.freeze([]);
17
-
18
- export class VercelRSCRuntime<T extends WeakKey = VercelRSCMessage>
19
- extends ProxyConfigProvider
20
- implements AssistantRuntime, ReactThreadRuntime
21
- {
22
- private useAdapter: UseBoundStore<StoreApi<{ adapter: VercelRSCAdapter<T> }>>;
23
-
24
- private _subscriptions = new Set<() => void>();
25
-
26
- public isRunning = false;
27
- public messages: ThreadMessage[] = [];
28
-
29
- constructor(public adapter: VercelRSCAdapter<T>) {
30
- super();
31
-
32
- this.useAdapter = create(() => ({
33
- adapter,
34
- }));
35
- }
36
-
37
- private withRunning = (callback: Promise<unknown>) => {
38
- this.isRunning = true;
39
- return callback.finally(() => {
40
- this.isRunning = false;
41
- });
42
- };
43
-
44
- public getBranches(): readonly string[] {
45
- return EMPTY_BRANCHES;
46
- }
47
-
48
- public switchToBranch(): void {
49
- throw new Error(
50
- "Branch switching is not supported by VercelRSCAssistantProvider.",
51
- );
52
- }
53
-
54
- public async append(message: AppendMessage): Promise<void> {
55
- if (message.parentId !== (this.messages.at(-1)?.id ?? null)) {
56
- if (!this.adapter.edit)
57
- throw new Error(
58
- "Message editing is not enabled, please provide an edit callback to VercelRSCAssistantProvider.",
59
- );
60
- await this.withRunning(this.adapter.edit(message));
61
- } else {
62
- await this.withRunning(this.adapter.append(message));
63
- }
64
- }
65
-
66
- public async startRun(parentId: string | null): Promise<void> {
67
- if (!this.adapter.reload)
68
- throw new Error(
69
- "Message reloading is not enabled, please provide a reload callback to VercelRSCAssistantProvider.",
70
- );
71
- await this.withRunning(this.adapter.reload(parentId));
72
- }
73
-
74
- cancelRun(): void {
75
- // in dev mode, log a warning
76
- if (process.env["NODE_ENV"] === "development") {
77
- console.warn(
78
- "Run cancellation is not supported by VercelRSCAssistantProvider.",
79
- );
80
- }
81
- }
82
-
83
- public subscribe(callback: () => void): Unsubscribe {
84
- this._subscriptions.add(callback);
85
- return () => this._subscriptions.delete(callback);
86
- }
87
-
88
- public onAdapterUpdated() {
89
- if (this.useAdapter.getState().adapter !== this.adapter) {
90
- this.useAdapter.setState({ adapter: this.adapter });
91
- }
92
- }
93
-
94
- private updateData = (messages: ThreadMessage[]) => {
95
- this.messages = messages;
96
- for (const callback of this._subscriptions) callback();
97
- };
98
-
99
- unstable_synchronizer = () => {
100
- const { adapter } = this.useAdapter();
101
-
102
- useVercelRSCSync(adapter, this.updateData);
103
-
104
- return null;
105
- };
106
- }
@@ -1,11 +0,0 @@
1
- import type { ThreadMessage } from "@assistant-ui/react";
2
-
3
- export const symbolInnerRSCMessage = Symbol("innerVercelRSCMessage");
4
-
5
- export type VercelRSCThreadMessage<T> = ThreadMessage & {
6
- [symbolInnerRSCMessage]?: T;
7
- };
8
-
9
- export const getVercelRSCMessage = <T,>(message: ThreadMessage) => {
10
- return (message as VercelRSCThreadMessage<T>)[symbolInnerRSCMessage];
11
- };
package/src/rsc/index.ts DELETED
@@ -1,4 +0,0 @@
1
- export { useVercelRSCRuntime } from "./useVercelRSCRuntime";
2
- export { getVercelRSCMessage } from "./getVercelRSCMessage";
3
- export type { VercelRSCAdapter } from "./VercelRSCAdapter";
4
- export type { VercelRSCMessage } from "./VercelRSCMessage";
@@ -1,20 +0,0 @@
1
- "use client";
2
-
3
- import { useEffect, useInsertionEffect, useState } from "react";
4
- import type { VercelRSCAdapter } from "./VercelRSCAdapter";
5
- import { VercelRSCRuntime } from "./VercelRSCRuntime";
6
-
7
- export const useVercelRSCRuntime = <T extends WeakKey>(
8
- adapter: VercelRSCAdapter<T>,
9
- ) => {
10
- const [runtime] = useState(() => new VercelRSCRuntime(adapter));
11
-
12
- useInsertionEffect(() => {
13
- runtime.adapter = adapter;
14
- });
15
- useEffect(() => {
16
- runtime.onAdapterUpdated();
17
- });
18
-
19
- return runtime;
20
- };
@@ -1,50 +0,0 @@
1
- import type { ThreadMessage } from "@assistant-ui/react";
2
- import { useEffect, useMemo } from "react";
3
- import {
4
- type ConverterCallback,
5
- ThreadMessageConverter,
6
- } from "../utils/ThreadMessageConverter";
7
- import type { VercelRSCAdapter } from "./VercelRSCAdapter";
8
- import type { VercelRSCMessage } from "./VercelRSCMessage";
9
- import {
10
- type VercelRSCThreadMessage,
11
- symbolInnerRSCMessage,
12
- } from "./getVercelRSCMessage";
13
-
14
- const vercelToThreadMessage = <T,>(
15
- converter: (message: T) => VercelRSCMessage,
16
- rawMessage: T,
17
- ): VercelRSCThreadMessage<T> => {
18
- const message = converter(rawMessage);
19
-
20
- return {
21
- id: message.id,
22
- role: message.role,
23
- content: [{ type: "ui", display: message.display }],
24
- createdAt: message.createdAt ?? new Date(),
25
- ...{ status: "done" },
26
- [symbolInnerRSCMessage]: rawMessage,
27
- };
28
- };
29
-
30
- type UpdateDataCallback = (messages: ThreadMessage[]) => void;
31
-
32
- export const useVercelRSCSync = <T extends WeakKey>(
33
- adapter: VercelRSCAdapter<T>,
34
- updateData: UpdateDataCallback,
35
- ) => {
36
- // flush the converter cache when the convertMessage prop changes
37
- const [converter, convertCallback] = useMemo(() => {
38
- const rscConverter =
39
- adapter.convertMessage ?? ((m: T) => m as VercelRSCMessage);
40
- const convertCallback: ConverterCallback<T> = (m, cache) => {
41
- if (cache) return cache;
42
- return vercelToThreadMessage(rscConverter, m);
43
- };
44
- return [new ThreadMessageConverter(), convertCallback];
45
- }, [adapter.convertMessage]);
46
-
47
- useEffect(() => {
48
- updateData(converter.convertMessages(adapter.messages, convertCallback));
49
- }, [updateData, converter, convertCallback, adapter.messages]);
50
- };
@@ -1,169 +0,0 @@
1
- import type {
2
- AssistantRuntime,
3
- ReactThreadRuntime,
4
- Unsubscribe,
5
- } from "@assistant-ui/react";
6
- import type { AppendMessage, ThreadMessage } from "@assistant-ui/react";
7
- import {
8
- MessageRepository,
9
- ProxyConfigProvider,
10
- } from "@assistant-ui/react/internal";
11
- import type { Message } from "ai";
12
- import { type StoreApi, type UseBoundStore, create } from "zustand";
13
- import { getVercelAIMessage } from "./getVercelAIMessage";
14
- import type { VercelHelpers } from "./utils/VercelHelpers";
15
- import { sliceMessagesUntil } from "./utils/sliceMessagesUntil";
16
- import { useVercelAIComposerSync } from "./utils/useVercelAIComposerSync";
17
- import { useVercelAIThreadSync } from "./utils/useVercelAIThreadSync";
18
-
19
- const hasUpcomingMessage = (isRunning: boolean, messages: ThreadMessage[]) => {
20
- return isRunning && messages[messages.length - 1]?.role !== "assistant";
21
- };
22
-
23
- export class VercelAIRuntime
24
- extends ProxyConfigProvider
25
- implements AssistantRuntime, ReactThreadRuntime
26
- {
27
- private _subscriptions = new Set<() => void>();
28
- private repository = new MessageRepository();
29
- private assistantOptimisticId: string | null = null;
30
-
31
- private useVercel: UseBoundStore<StoreApi<{ vercel: VercelHelpers }>>;
32
-
33
- public messages: ThreadMessage[] = [];
34
- public isRunning = false;
35
-
36
- constructor(public vercel: VercelHelpers) {
37
- super();
38
- this.useVercel = create(() => ({
39
- vercel,
40
- }));
41
- }
42
-
43
- public getBranches(messageId: string): string[] {
44
- return this.repository.getBranches(messageId);
45
- }
46
-
47
- public switchToBranch(branchId: string): void {
48
- this.repository.switchToBranch(branchId);
49
- this.updateVercelMessages(this.repository.getMessages());
50
- }
51
-
52
- public async append(message: AppendMessage): Promise<void> {
53
- // add user message
54
- if (message.content.length !== 1 || message.content[0]?.type !== "text")
55
- throw new Error("Only text content is supported by Vercel AI SDK.");
56
-
57
- const newMessages = sliceMessagesUntil(
58
- this.vercel.messages,
59
- message.parentId,
60
- );
61
- this.vercel.setMessages(newMessages);
62
-
63
- await this.vercel.append({
64
- role: "user",
65
- content: message.content[0].text,
66
- });
67
- }
68
-
69
- public async startRun(parentId: string | null): Promise<void> {
70
- const reloadMaybe =
71
- "reload" in this.vercel ? this.vercel.reload : undefined;
72
- if (!reloadMaybe)
73
- throw new Error(
74
- "Reload is not supported by Vercel AI SDK's useAssistant.",
75
- );
76
-
77
- const newMessages = sliceMessagesUntil(this.vercel.messages, parentId);
78
- this.vercel.setMessages(newMessages);
79
-
80
- await reloadMaybe();
81
- }
82
-
83
- public cancelRun(): void {
84
- const previousMessage = this.vercel.messages.at(-1);
85
-
86
- this.vercel.stop();
87
-
88
- if (this.assistantOptimisticId) {
89
- this.repository.deleteMessage(this.assistantOptimisticId);
90
- this.assistantOptimisticId = null;
91
- }
92
-
93
- let messages = this.repository.getMessages();
94
- if (
95
- previousMessage?.role === "user" &&
96
- previousMessage.id === messages.at(-1)?.id // ensure the previous message is a leaf node
97
- ) {
98
- this.vercel.setInput(previousMessage.content);
99
- this.repository.deleteMessage(previousMessage.id);
100
-
101
- messages = this.repository.getMessages();
102
- }
103
-
104
- // resync messages
105
- setTimeout(() => {
106
- this.updateVercelMessages(messages);
107
- }, 0);
108
- }
109
-
110
- public subscribe(callback: () => void): Unsubscribe {
111
- this._subscriptions.add(callback);
112
- return () => this._subscriptions.delete(callback);
113
- }
114
-
115
- private updateVercelMessages = (messages: ThreadMessage[]) => {
116
- this.vercel.setMessages(
117
- messages
118
- .flatMap(getVercelAIMessage)
119
- .filter((m): m is Message => m != null),
120
- );
121
- };
122
-
123
- public onVercelUpdated() {
124
- if (this.useVercel.getState().vercel !== this.vercel) {
125
- this.useVercel.setState({ vercel: this.vercel });
126
- }
127
- }
128
-
129
- private updateData = (isRunning: boolean, vm: ThreadMessage[]) => {
130
- for (let i = 0; i < vm.length; i++) {
131
- const message = vm[i]!;
132
- const parent = vm[i - 1];
133
- this.repository.addOrUpdateMessage(parent?.id ?? null, message);
134
- }
135
-
136
- if (this.assistantOptimisticId) {
137
- this.repository.deleteMessage(this.assistantOptimisticId);
138
- this.assistantOptimisticId = null;
139
- }
140
-
141
- if (hasUpcomingMessage(isRunning, vm)) {
142
- this.assistantOptimisticId = this.repository.appendOptimisticMessage(
143
- vm.at(-1)?.id ?? null,
144
- {
145
- role: "assistant",
146
- content: [{ type: "text", text: "" }],
147
- },
148
- );
149
- }
150
-
151
- this.repository.resetHead(
152
- this.assistantOptimisticId ?? vm.at(-1)?.id ?? null,
153
- );
154
-
155
- this.messages = this.repository.getMessages();
156
- this.isRunning = isRunning;
157
-
158
- for (const callback of this._subscriptions) callback();
159
- };
160
-
161
- unstable_synchronizer = () => {
162
- const { vercel } = this.useVercel();
163
-
164
- useVercelAIThreadSync(vercel, this.updateData);
165
- useVercelAIComposerSync(vercel);
166
-
167
- return null;
168
- };
169
- }
@@ -1,12 +0,0 @@
1
- import type { ThreadMessage } from "@assistant-ui/react";
2
- import type { Message } from "ai";
3
-
4
- export const symbolInnerAIMessage = Symbol("innerVercelAIUIMessage");
5
-
6
- export type VercelAIThreadMessage = ThreadMessage & {
7
- [symbolInnerAIMessage]?: Message[];
8
- };
9
-
10
- export const getVercelAIMessage = (message: ThreadMessage) => {
11
- return (message as VercelAIThreadMessage)[symbolInnerAIMessage];
12
- };
package/src/ui/index.ts DELETED
@@ -1,3 +0,0 @@
1
- export { useVercelUseChatRuntime } from "./use-chat/useVercelUseChatRuntime";
2
- export { useVercelUseAssistantRuntime } from "./use-assistant/useVercelUseAssistantRuntime";
3
- export { getVercelAIMessage } from "./getVercelAIMessage";
@@ -1,18 +0,0 @@
1
- import type { UseAssistantHelpers } from "@ai-sdk/react";
2
- import { useEffect, useInsertionEffect, useState } from "react";
3
- import { VercelAIRuntime } from "../VercelAIRuntime";
4
-
5
- export const useVercelUseAssistantRuntime = (
6
- assistantHelpers: UseAssistantHelpers,
7
- ) => {
8
- const [runtime] = useState(() => new VercelAIRuntime(assistantHelpers));
9
-
10
- useInsertionEffect(() => {
11
- runtime.vercel = assistantHelpers;
12
- });
13
- useEffect(() => {
14
- runtime.onVercelUpdated();
15
- });
16
-
17
- return runtime;
18
- };