Bring your own redux, zustand, or state manager.
ExternalStoreRuntime bridges your existing state management with assistant-ui. You provide messages and callbacks; the runtime renders whatever you give it. UI features turn on based on which callbacks are present.
When to use it
Pick ExternalStoreRuntime when:
- You already keep messages in redux, zustand, tanstack-query, or another store, and want to keep them there.
- You want full control over message state, persistence, and synchronization.
- You have a custom message format and need automatic conversion to assistant-ui's format.
If you do not have an existing store, use LocalRuntime instead; it is lower-friction.
Architecture
Key idea: you own the state, the adapter translates between your format and assistant-ui's. UI features are capability-based; if you provide setMessages, branching turns on; if you provide onEdit, editing turns on; etc.
Quickstart
Install
npm install @assistant-ui/reactCreate the runtime provider
"use client";
import { useState, ReactNode } from "react";
import {
useExternalStoreRuntime,
ThreadMessageLike,
AppendMessage,
AssistantRuntimeProvider,
} from "@assistant-ui/react";
type MyMessage = { role: "user" | "assistant"; content: string };
const convertMessage = (message: MyMessage): ThreadMessageLike => ({
role: message.role,
content: [{ type: "text", text: message.content }],
});
const backendApi = async (input: string): Promise<MyMessage> => {
return { role: "assistant", content: "Hello, world!" };
};
export function MyRuntimeProvider({
children,
}: Readonly<{ children: ReactNode }>) {
const [isRunning, setIsRunning] = useState(false);
const [messages, setMessages] = useState<MyMessage[]>([]);
const onNew = async (message: AppendMessage) => {
if (message.content[0]?.type !== "text") {
throw new Error("Only text messages are supported");
}
const input = message.content[0].text;
setMessages((prev) => [...prev, { role: "user", content: input }]);
setIsRunning(true);
const assistant = await backendApi(input);
setMessages((prev) => [...prev, assistant]);
setIsRunning(false);
};
const runtime = useExternalStoreRuntime({
isRunning,
messages,
convertMessage,
onNew,
});
return (
<AssistantRuntimeProvider runtime={runtime}>
{children}
</AssistantRuntimeProvider>
);
}Use in your app
import { Thread } from "@/components/assistant-ui/thread";
import { MyRuntimeProvider } from "./MyRuntimeProvider";
export default function Page() {
return (
<MyRuntimeProvider>
<Thread />
</MyRuntimeProvider>
);
}Message conversion
Two approaches.
Inline convertMessage
const convertMessage = (message: MyMessage): ThreadMessageLike => ({
role: message.role,
content: [{ type: "text", text: message.text }],
id: message.id,
createdAt: new Date(message.timestamp),
});
const runtime = useExternalStoreRuntime({
messages: myMessages,
convertMessage,
onNew,
});useExternalMessageConverter (with join strategy)
For performance optimization or when you need to merge adjacent assistant messages:
import { useExternalMessageConverter } from "@assistant-ui/react";
const convertedMessages = useExternalMessageConverter({
callback: (message: MyMessage): ThreadMessageLike => ({
role: message.role,
content: [{ type: "text", text: message.text }],
id: message.id,
}),
messages,
isRunning: false,
joinStrategy: "concat-content", // merges adjacent assistant messages
});
const runtime = useExternalStoreRuntime({
messages: convertedMessages,
onNew,
});joinStrategy controls how adjacent assistant messages combine: concat-content (default) merges them into one; none keeps them separate.
Handler matrix
Each handler enables a specific UI feature.
| Handler | Enables |
|---|---|
onNew | Sending new user messages (required) |
setMessages | Branch switching |
onEdit | Message edit button |
onReload | Regenerate button |
onCancel | Cancel button while generating |
onAddToolResult | Client-side tool result handoff |
Streaming responses
Stream by mutating the assistant message in place:
const onNew = async (message: AppendMessage) => {
const userMsg: ThreadMessageLike = {
role: "user",
content: message.content,
id: generateId(),
};
setMessages((prev) => [...prev, userMsg]);
setIsRunning(true);
const assistantId = generateId();
setMessages((prev) => [
...prev,
{ role: "assistant", content: [{ type: "text", text: "" }], id: assistantId },
]);
const stream = await api.streamChat(message);
for await (const chunk of stream) {
setMessages((prev) =>
prev.map((m) =>
m.id === assistantId
? {
...m,
content: [
{ type: "text", text: (m.content[0] as any).text + chunk },
],
}
: m,
),
);
}
setIsRunning(false);
};Message editing
const onEdit = async (message: AppendMessage) => {
const index = messages.findIndex((m) => m.id === message.parentId) + 1;
const newMessages = [...messages.slice(0, index)];
newMessages.push({
role: "user",
content: message.content,
id: message.id ?? generateId(),
});
setMessages(newMessages);
setIsRunning(true);
const response = await api.chat(message);
newMessages.push({ role: "assistant", content: response.content, id: generateId() });
setMessages(newMessages);
setIsRunning(false);
};Branching
The linear messages array assumes each message's parent is the previous one. For branching (e.g. multiple regenerations), use ExportedMessageRepository.fromBranchableArray() and import via thread.import():
import {
ExportedMessageRepository,
useExternalStoreRuntime,
} from "@assistant-ui/react";
const backendMessages = [
{ id: "user-1", role: "user", content: "Hello", parentId: null },
{ id: "asst-1", role: "assistant", content: "Hi!", parentId: "user-1" },
{ id: "asst-2", role: "assistant", content: "Hey!", parentId: "user-1" }, // branch
];
const repo = ExportedMessageRepository.fromBranchableArray(
backendMessages.map((m) => ({
message: { id: m.id, role: m.role, content: m.content },
parentId: m.parentId,
})),
{ headId: "asst-1" },
);
runtime.thread.import(repo);Each message must have an explicit id and parentId; messages with the same parentId create branches. Parents must appear before children in the array.
Tool calling
Handle tool results by updating the matching tool-call entry:
const onAddToolResult = (options: AddToolResultOptions) => {
setMessages((prev) =>
prev.map((message) =>
message.id === options.messageId
? {
...message,
content: message.content.map((part) =>
part.type === "tool-call" &&
part.toolCallId === options.toolCallId
? { ...part, result: options.result }
: part,
),
}
: message,
),
);
};
const runtime = useExternalStoreRuntime({
messages,
onNew,
onAddToolResult,
});The runtime automatically matches tool results to their tool calls by toolCallId and groups related messages for display.
Attachments
Attachments use the standard adapter contract, see adapters:
const runtime = useExternalStoreRuntime({
messages,
onNew,
adapters: { attachments: myAttachmentAdapter },
});Multi-thread
ExternalStoreRuntime uses ExternalStoreThreadListAdapter (synchronous, inline). See threads for the contract and best practices on keeping currentThreadId in sync with your store.
Integration examples
Redux
import { createSlice, PayloadAction } from "@reduxjs/toolkit";
import { ThreadMessageLike } from "@assistant-ui/react";
const chatSlice = createSlice({
name: "chat",
initialState: { messages: [] as ThreadMessageLike[], isRunning: false },
reducers: {
setMessages: (state, action: PayloadAction<ThreadMessageLike[]>) => {
state.messages = action.payload;
},
addMessage: (state, action: PayloadAction<ThreadMessageLike>) => {
state.messages.push(action.payload);
},
setIsRunning: (state, action: PayloadAction<boolean>) => {
state.isRunning = action.payload;
},
},
});
export const { setMessages, addMessage, setIsRunning } = chatSlice.actions;import { useSelector, useDispatch } from "react-redux";
import { useExternalStoreRuntime, AssistantRuntimeProvider } from "@assistant-ui/react";
export function ReduxRuntimeProvider({ children }) {
const messages = useSelector((s: RootState) => s.chat.messages);
const isRunning = useSelector((s: RootState) => s.chat.isRunning);
const dispatch = useDispatch();
const runtime = useExternalStoreRuntime({
messages,
isRunning,
setMessages: (messages) => dispatch(setMessages(messages)),
onNew: async (message) => {
dispatch(
addMessage({
role: "user",
content: message.content,
id: `msg-${Date.now()}`,
createdAt: new Date(),
}),
);
dispatch(setIsRunning(true));
const response = await api.chat(message);
dispatch(
addMessage({
role: "assistant",
content: response.content,
id: `msg-${Date.now()}`,
createdAt: new Date(),
}),
);
dispatch(setIsRunning(false));
},
});
return (
<AssistantRuntimeProvider runtime={runtime}>
{children}
</AssistantRuntimeProvider>
);
}Zustand
import { create } from "zustand";
import { immer } from "zustand/middleware/immer";
import { ThreadMessageLike } from "@assistant-ui/react";
interface ChatState {
messages: ThreadMessageLike[];
isRunning: boolean;
addMessage: (message: ThreadMessageLike) => void;
setMessages: (messages: ThreadMessageLike[]) => void;
setIsRunning: (isRunning: boolean) => void;
}
export const useChatStore = create<ChatState>()(
immer((set) => ({
messages: [],
isRunning: false,
addMessage: (message) => set((s) => { s.messages.push(message); }),
setMessages: (messages) => set((s) => { s.messages = messages; }),
setIsRunning: (isRunning) => set((s) => { s.isRunning = isRunning; }),
})),
);import { useShallow } from "zustand/shallow";
import { useExternalStoreRuntime, AssistantRuntimeProvider } from "@assistant-ui/react";
export function ZustandRuntimeProvider({ children }) {
const { messages, isRunning, addMessage, setMessages, setIsRunning } =
useChatStore(
useShallow((s) => ({
messages: s.messages,
isRunning: s.isRunning,
addMessage: s.addMessage,
setMessages: s.setMessages,
setIsRunning: s.setIsRunning,
})),
);
const runtime = useExternalStoreRuntime({
messages,
isRunning,
setMessages,
onNew: async (message) => {
addMessage({
role: "user",
content: message.content,
id: `msg-${Date.now()}`,
createdAt: new Date(),
});
setIsRunning(true);
const response = await api.chat(message);
addMessage({
role: "assistant",
content: response.content,
id: `msg-${Date.now()}-a`,
createdAt: new Date(),
});
setIsRunning(false);
},
});
return (
<AssistantRuntimeProvider runtime={runtime}>
{children}
</AssistantRuntimeProvider>
);
}TanStack Query
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
import { useExternalStoreRuntime } from "@assistant-ui/react";
const messageKeys = {
all: ["messages"] as const,
thread: (threadId: string) => [...messageKeys.all, threadId] as const,
};
export function TanStackQueryRuntimeProvider({ children }) {
const queryClient = useQueryClient();
const threadId = "main";
const { data: messages = [] } = useQuery({
queryKey: messageKeys.thread(threadId),
queryFn: () => fetchMessages(threadId),
});
const sendMessage = useMutation({
mutationFn: api.chat,
onMutate: async (message: AppendMessage) => {
await queryClient.cancelQueries({
queryKey: messageKeys.thread(threadId),
});
const previous = queryClient.getQueryData<ThreadMessageLike[]>(
messageKeys.thread(threadId),
);
queryClient.setQueryData<ThreadMessageLike[]>(
messageKeys.thread(threadId),
(old = []) => [
...old,
{
role: "user",
content: message.content,
id: `temp-${Date.now()}`,
createdAt: new Date(),
},
],
);
return { previous };
},
onError: (_err, _msg, context) => {
if (context?.previous) {
queryClient.setQueryData(messageKeys.thread(threadId), context.previous);
}
},
onSettled: () =>
queryClient.invalidateQueries({ queryKey: messageKeys.thread(threadId) }),
});
const runtime = useExternalStoreRuntime({
messages,
isRunning: sendMessage.isPending,
onNew: async (message) => {
await sendMessage.mutateAsync(message);
},
setMessages: (newMessages) => {
queryClient.setQueryData(messageKeys.thread(threadId), newMessages);
},
});
return (
<AssistantRuntimeProvider runtime={runtime}>
{children}
</AssistantRuntimeProvider>
);
}Working with external messages
getExternalStoreMessages
Retrieve your original message format from any assistant-ui state:
import { getExternalStoreMessages, useAuiState } from "@assistant-ui/react";
function MyComponent() {
const originalMessages = useAuiState((s) => getExternalStoreMessages(s.message));
// originalMessages is MyMessage[] (your original type)
}getExternalStoreMessages may return multiple messages for a single UI message; assistant-ui merges adjacent assistant and tool messages for display.
bindExternalStoreMessage
Attach your original message to a ThreadMessage you constructed manually (outside the built-in converter):
import {
bindExternalStoreMessage,
getExternalStoreMessages,
} from "@assistant-ui/react";
bindExternalStoreMessage(threadMessage, originalMessage);
const original = getExternalStoreMessages(threadMessage);bindExternalStoreMessage is a no-op if the target already has a bound message. It mutates the target in place.
This API is experimental and may change without notice.
Best practices
- Immutable updates. Always create new arrays:
setMessages([...messages, newMessage]); // not messages.push(newMessage) - Stable handler references. Memoize
onNew,onEdit, etc. withuseCallbackto avoid recreating the runtime. - Use
useShallowwith zustand to prevent unnecessary re-renders.
Common pitfalls
Edit / regenerate / cancel buttons missing. Each requires its handler:
useExternalStoreRuntime({
messages,
onNew, // required
setMessages, // branch switching
onEdit, // edit
onReload, // regenerate
onCancel, // cancel
});State not updating. check for: array mutation instead of new arrays, missing setMessages, broken async handling, or invalid convertMessage output.
Messages going to the wrong thread. the runtime's currentThreadId and your store's selected thread must stay in sync. Centralize thread id in a context, never in component-local state. See threads.
API reference
ExternalStoreAdapter
ExternalStoreAdapter<T>messages: readonly T[]Array of messages from your state.
onNew: (message: AppendMessage) => Promise<void>Handler for new messages from the user.
isRunning: boolean= falseWhether the assistant is currently generating a response. When true, shows an optimistic assistant message and flows directly to thread.isRunning.
isDisabled: boolean= falseWhether the chat input should be disabled.
isLoading?: booleanWhether the adapter is in a loading state. Displays a loading indicator instead of the composer.
suggestions?: readonly ThreadSuggestion[]Suggested prompts to display.
extras?: unknownAdditional data accessible via runtime.extras.
setMessages?: (messages: readonly T[]) => voidUpdate messages (required for branch switching).
onEdit?: (message: AppendMessage) => Promise<void>Handler for message edits (required for edit feature).
onReload?: (parentId: string | Null, config: StartRunConfig) => Promise<void>Handler for regenerating messages (required for reload feature).
onCancel?: () => Promise<void>Handler for cancelling the current generation.
onAddToolResult?: (options: AddToolResultOptions) => Promise<void> | VoidHandler for adding tool call results.
onResume?: (config: ResumeRunConfig) => Promise<void>Handler for resuming an interrupted run (e.g. after a page reload mid-generation).
onResumeToolCall?: (options: { toolCallId: string; payload: unknown }) => voidHandler for resuming a suspended tool call (used with human-in-the-loop tool execution).
messageRepository?: ExportedMessageRepositoryPre-built message repository with branching history. Use instead of messages when you need to restore branch state.
state?: ReadonlyJSONValueOpaque serializable state passed to onLoadExternalState during thread import.
onImport?: (messages: readonly ThreadMessage[]) => voidCalled when the runtime imports messages into the external store (e.g. on thread switch).
onExportExternalState?: () => anyCalled to retrieve external state when the runtime exports a thread snapshot.
onLoadExternalState?: (state: any) => voidCalled with previously exported external state when restoring a thread snapshot.
convertMessage?: (message: T, index: number) => ThreadMessageLikeConvert your message format to assistant-ui format. Not needed if using ThreadMessage type.
adapters?: objectCapability adapters: attachments, speech, dictation, feedback, threadList. See /docs/runtimes/concepts/adapters.
unstable_capabilitiesunstable?: objectConfigure runtime capabilities (e.g. copy). Unstable, may change.
ThreadMessageLike
ThreadMessageLikerole: "assistant" | "user" | "system"The role of the message sender.
content: string | Readonly MessagePart[]Message content as string or structured message parts. Supports data-* prefixed types (e.g. { type: "data-workflow", data: {...} }) which are automatically converted to DataMessagePart.
id?: stringUnique identifier for the message.
createdAt?: DateTimestamp when the message was created.
status?: MessageStatusStatus of assistant messages ({ type: "running" }, { type: "complete" }, { type: "incomplete" }).
attachments?: readonly CompleteAttachment[]File attachments (user messages only). Type accepts custom strings beyond "image" | "document" | "file"; contentType is optional.
metadata?: objectAdditional message metadata (steps, custom fields).