@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.
@@ -1,16 +0,0 @@
1
- import type { UseChatHelpers } from "@ai-sdk/react";
2
- import { useEffect, useInsertionEffect, useState } from "react";
3
- import { VercelAIRuntime } from "../VercelAIRuntime";
4
-
5
- export const useVercelUseChatRuntime = (chatHelpers: UseChatHelpers) => {
6
- const [runtime] = useState(() => new VercelAIRuntime(chatHelpers));
7
-
8
- useInsertionEffect(() => {
9
- runtime.vercel = chatHelpers;
10
- });
11
- useEffect(() => {
12
- runtime.onVercelUpdated();
13
- });
14
-
15
- return runtime;
16
- };
@@ -1,3 +0,0 @@
1
- import type { UseAssistantHelpers, UseChatHelpers } from "@ai-sdk/react";
2
-
3
- export type VercelHelpers = UseChatHelpers | UseAssistantHelpers;
@@ -1,20 +0,0 @@
1
- import type { Message } from "ai";
2
-
3
- export const sliceMessagesUntil = (
4
- messages: Message[],
5
- messageId: string | null,
6
- ) => {
7
- if (messageId == null) return [];
8
-
9
- let messageIdx = messages.findIndex((m) => m.id === messageId);
10
- if (messageIdx === -1)
11
- throw new Error(
12
- "useVercelAIThreadState: Message not found. This is liekly an internal bug in assistant-ui.",
13
- );
14
-
15
- while (messages[messageIdx + 1]?.role === "assistant") {
16
- messageIdx++;
17
- }
18
-
19
- return messages.slice(0, messageIdx + 1);
20
- };
@@ -1,15 +0,0 @@
1
- import { useThreadContext } from "@assistant-ui/react/experimental";
2
- import { useEffect } from "react";
3
- import type { VercelHelpers } from "./VercelHelpers";
4
-
5
- // two way sync between vercel helpers input state and composer text state
6
- export const useVercelAIComposerSync = (vercel: VercelHelpers) => {
7
- const { useComposer } = useThreadContext();
8
-
9
- useEffect(() => {
10
- useComposer.setState({
11
- value: vercel.input,
12
- setValue: vercel.setInput,
13
- });
14
- }, [useComposer, vercel.input, vercel.setInput]);
15
- };
@@ -1,142 +0,0 @@
1
- import type { TextContentPart, ThreadMessage } from "@assistant-ui/react";
2
- import type { ToolCallContentPart } from "@assistant-ui/react/experimental";
3
- import type { Message } from "ai";
4
- import { useEffect, useMemo } from "react";
5
- import {
6
- type ConverterCallback,
7
- ThreadMessageConverter,
8
- } from "../../utils/ThreadMessageConverter";
9
- import {
10
- type VercelAIThreadMessage,
11
- symbolInnerAIMessage,
12
- } from "../getVercelAIMessage";
13
- import type { VercelHelpers } from "./VercelHelpers";
14
-
15
- const getIsRunning = (vercel: VercelHelpers) => {
16
- if ("isLoading" in vercel) return vercel.isLoading;
17
- return vercel.status === "in_progress";
18
- };
19
-
20
- const vercelToThreadMessage = (
21
- messages: Message[],
22
- status: "in_progress" | "done" | "error",
23
- ): VercelAIThreadMessage => {
24
- const firstMessage = messages[0];
25
- if (!firstMessage) throw new Error("No messages found");
26
-
27
- const common = {
28
- id: firstMessage.id,
29
- createdAt: firstMessage.createdAt ?? new Date(),
30
- [symbolInnerAIMessage]: messages,
31
- };
32
-
33
- switch (firstMessage.role) {
34
- case "user":
35
- if (messages.length > 1) {
36
- throw new Error(
37
- "Multiple user messages found. This is likely an internal bug in assistant-ui.",
38
- );
39
- }
40
-
41
- return {
42
- ...common,
43
- role: "user",
44
- content: [{ type: "text", text: firstMessage.content }],
45
- };
46
- case "assistant":
47
- return {
48
- ...common,
49
- role: "assistant",
50
- content: messages.flatMap((message) => [
51
- ...(message.content
52
- ? [{ type: "text", text: message.content } as TextContentPart]
53
- : []),
54
- ...(message.toolInvocations?.map(
55
- (t) =>
56
- ({
57
- type: "tool-call",
58
- name: t.toolName,
59
- args: t.args,
60
- result: "result" in t ? t.result : undefined,
61
- }) as ToolCallContentPart,
62
- ) ?? []),
63
- ]),
64
- status,
65
- };
66
- default:
67
- throw new Error(
68
- `You have a message with an unsupported role. The role ${firstMessage.role} is not supported.`,
69
- );
70
- }
71
- };
72
-
73
- type Chunk = [Message, ...Message[]];
74
- const hasItems = (messages: Message[]): messages is Chunk =>
75
- messages.length > 0;
76
-
77
- const chunkedMessages = (messages: Message[]): Chunk[] => {
78
- const chunks: Chunk[] = [];
79
- let currentChunk: Message[] = [];
80
-
81
- for (const message of messages) {
82
- if (message.role === "assistant") {
83
- currentChunk.push(message);
84
- } else {
85
- if (hasItems(currentChunk)) {
86
- chunks.push(currentChunk);
87
- currentChunk = [];
88
- }
89
- chunks.push([message]);
90
- }
91
- }
92
-
93
- if (hasItems(currentChunk)) {
94
- chunks.push(currentChunk);
95
- }
96
-
97
- return chunks;
98
- };
99
-
100
- const shallowArrayEqual = (a: unknown[], b: unknown[]) => {
101
- if (a.length !== b.length) return false;
102
- for (let i = 0; i < a.length; i++) {
103
- if (a[i] !== b[i]) return false;
104
- }
105
- return true;
106
- };
107
-
108
- type UpdateDataCallback = (isRunning: boolean, vm: ThreadMessage[]) => void;
109
-
110
- export const useVercelAIThreadSync = (
111
- vercel: VercelHelpers,
112
- updateData: UpdateDataCallback,
113
- ) => {
114
- const isRunning = getIsRunning(vercel);
115
-
116
- const converter = useMemo(() => new ThreadMessageConverter(), []);
117
-
118
- useEffect(() => {
119
- const lastMessageId = vercel.messages.at(-1)?.id;
120
- const convertCallback: ConverterCallback<Chunk> = (messages, cache) => {
121
- const status =
122
- lastMessageId === messages[0].id && isRunning ? "in_progress" : "done";
123
-
124
- if (
125
- cache &&
126
- shallowArrayEqual(cache.content, messages) &&
127
- (cache.role === "user" || cache.status === status)
128
- )
129
- return cache;
130
-
131
- return vercelToThreadMessage(messages, status);
132
- };
133
-
134
- const messages = converter.convertMessages(
135
- chunkedMessages(vercel.messages),
136
- convertCallback,
137
- (m) => m[0],
138
- );
139
-
140
- updateData(isRunning, messages);
141
- }, [updateData, isRunning, vercel.messages, converter]);
142
- };
@@ -1,24 +0,0 @@
1
- import type { ThreadMessage } from "@assistant-ui/react";
2
-
3
- export type ConverterCallback<TIn> = (
4
- message: TIn,
5
- cache: ThreadMessage | undefined,
6
- ) => ThreadMessage;
7
-
8
- export class ThreadMessageConverter {
9
- private readonly cache = new WeakMap<WeakKey, ThreadMessage>();
10
-
11
- convertMessages<TIn extends WeakKey>(
12
- messages: TIn[],
13
- converter: ConverterCallback<TIn>,
14
- keyMapper: (m: TIn) => WeakKey = (key) => key,
15
- ): ThreadMessage[] {
16
- return messages.map((m) => {
17
- const key = keyMapper(m);
18
- const cached = this.cache.get(key);
19
- const newMessage = converter(m, cached);
20
- this.cache.set(key, newMessage);
21
- return newMessage;
22
- });
23
- }
24
- }
package/tsconfig.json DELETED
@@ -1,11 +0,0 @@
1
- {
2
- "extends": "@assistant-ui/tsconfig/base.json",
3
- "compilerOptions": {
4
- "paths": {
5
- "@assistant-ui/*": ["../../packages/*/src"],
6
- "@assistant-ui/react/*": ["../../packages/react/src/*"]
7
- }
8
- },
9
- "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx"],
10
- "exclude": ["node_modules"]
11
- }