Skip to content

Commit 14bb869

Browse files
authoredMay 31, 2024··
chore (ui): move maxAutomaticRoundtrips and addToolResult out of experimental (#1778)
1 parent f3267d5 commit 14bb869

File tree

5 files changed

+104
-83
lines changed

5 files changed

+104
-83
lines changed
 

‎.changeset/silly-pianos-speak.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'ai': patch
3+
---
4+
5+
chore (ui): move maxAutomaticRoundtrips and addToolResult out of experimental

‎content/docs/05-ai-sdk-ui/03-chatbot-with-tool-calling.mdx

+25-25
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ The flow is as follows:
2929
You can return the tool result from the callback.
3030
1. Client-side tool that require user interactions can be displayed in the UI.
3131
The tool calls and results are available in the `toolInvocations` property of the last assistant message.
32-
1. When the user interaction is done, `experimental_addToolResult` can be used to add the tool result to the chat.
32+
1. When the user interaction is done, `addToolResult` can be used to add the tool result to the chat.
3333
1. When there are tool calls in the last assistant message and all tool results are available, the client sends the updated messages back to the server.
3434
This triggers another iteration of this flow.
3535

@@ -39,9 +39,9 @@ The tool result contains all information about the tool call as well as the resu
3939

4040
<Note>
4141
In order to automatically send another request to the server when all tool
42-
calls are server-side, you need to set `experimental_maxAutomaticRoundtrips`
43-
to a value greater than 0 in the `useChat` options. It is disabled by default
44-
for backward compatibility.
42+
calls are server-side, you need to set `maxAutomaticRoundtrips` to a value
43+
greater than 0 in the `useChat` options. It is disabled by default for
44+
backward compatibility.
4545
</Note>
4646

4747
## Example
@@ -113,9 +113,9 @@ There are three things worth mentioning:
113113
1. The `toolInvocations` property of the last assistant message contains all tool calls and results.
114114
The client-side tool `askForConfirmation` is displayed in the UI.
115115
It asks the user for confirmation and displays the result once the user confirms or denies the execution.
116-
The result is added to the chat using `experimental_addToolResult`.
116+
The result is added to the chat using `addToolResult`.
117117

118-
1. The `experimental_maxAutomaticRoundtrips` option is set to 5.
118+
1. The `maxAutomaticRoundtrips` option is set to 5.
119119
This enables several tool use iterations between the client and the server.
120120

121121
```tsx filename='app/page.tsx'
@@ -125,23 +125,23 @@ import { ToolInvocation } from 'ai';
125125
import { Message, useChat } from 'ai/react';
126126

127127
export default function Chat() {
128-
const {
129-
messages,
130-
input,
131-
handleInputChange,
132-
handleSubmit,
133-
experimental_addToolResult,
134-
} = useChat({
135-
experimental_maxAutomaticRoundtrips: 5,
136-
137-
// run client-side tools that are automatically executed:
138-
async onToolCall({ toolCall }) {
139-
if (toolCall.toolName === 'getLocation') {
140-
const cities = ['New York', 'Los Angeles', 'Chicago', 'San Francisco'];
141-
return cities[Math.floor(Math.random() * cities.length)];
142-
}
143-
},
144-
});
128+
const { messages, input, handleInputChange, handleSubmit, addToolResult } =
129+
useChat({
130+
maxAutomaticRoundtrips: 5,
131+
132+
// run client-side tools that are automatically executed:
133+
async onToolCall({ toolCall }) {
134+
if (toolCall.toolName === 'getLocation') {
135+
const cities = [
136+
'New York',
137+
'Los Angeles',
138+
'Chicago',
139+
'San Francisco',
140+
];
141+
return cities[Math.floor(Math.random() * cities.length)];
142+
}
143+
},
144+
});
145145

146146
return (
147147
<div>
@@ -164,7 +164,7 @@ export default function Chat() {
164164
<>
165165
<button
166166
onClick={() =>
167-
experimental_addToolResult({
167+
addToolResult({
168168
toolCallId,
169169
result: 'Yes, confirmed.',
170170
})
@@ -174,7 +174,7 @@ export default function Chat() {
174174
</button>
175175
<button
176176
onClick={() =>
177-
experimental_addToolResult({
177+
addToolResult({
178178
toolCallId,
179179
result: 'No, denied',
180180
})

‎content/docs/07-reference/ai-sdk-ui/01-use-chat.mdx

+2-2
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,7 @@ Allows you to easily create a conversational user interface for your chatbot app
105105
"An optional boolean that determines whether to send extra fields you've added to `messages`. Defaults to `false` and only the `content` and `role` fields will be sent to the API endpoint.",
106106
},
107107
{
108-
name: 'experimental_maxAutomaticRoundtrips',
108+
name: 'maxAutomaticRoundtrips',
109109
type: 'number',
110110
description:
111111
'React only. Maximal number of automatic roundtrips for tool calls. An automatic tool call roundtrip is a call to the server with the tool call results when all tool calls in the last assistant message have results. A maximum number is required to prevent infinite loops in the case of misconfigured tools. By default, it is set to 0, which will disable the feature.',
@@ -184,7 +184,7 @@ Allows you to easily create a conversational user interface for your chatbot app
184184
description: 'Data returned from experimental_StreamData',
185185
},
186186
{
187-
name: 'experimental_addToolResult',
187+
name: 'addToolResult',
188188
type: '({toolCallId: string; result: any;}) => void',
189189
description:
190190
'React only. Function to add a tool result to the chat. This will update the chat messages with the tool result and call the API route if all tool results for the last message are available.',

‎examples/next-openai/app/use-chat-tools/page.tsx

+19-19
Original file line numberDiff line numberDiff line change
@@ -4,24 +4,24 @@ import { ToolInvocation } from 'ai';
44
import { Message, useChat } from 'ai/react';
55

66
export default function Chat() {
7-
const {
8-
messages,
9-
input,
10-
handleInputChange,
11-
handleSubmit,
12-
experimental_addToolResult,
13-
} = useChat({
14-
api: '/api/use-chat-tools',
15-
experimental_maxAutomaticRoundtrips: 5,
7+
const { messages, input, handleInputChange, handleSubmit, addToolResult } =
8+
useChat({
9+
api: '/api/use-chat-tools',
10+
maxAutomaticRoundtrips: 5,
1611

17-
// run client-side tools that are automatically executed:
18-
async onToolCall({ toolCall }) {
19-
if (toolCall.toolName === 'getLocation') {
20-
const cities = ['New York', 'Los Angeles', 'Chicago', 'San Francisco'];
21-
return cities[Math.floor(Math.random() * cities.length)];
22-
}
23-
},
24-
});
12+
// run client-side tools that are automatically executed:
13+
async onToolCall({ toolCall }) {
14+
if (toolCall.toolName === 'getLocation') {
15+
const cities = [
16+
'New York',
17+
'Los Angeles',
18+
'Chicago',
19+
'San Francisco',
20+
];
21+
return cities[Math.floor(Math.random() * cities.length)];
22+
}
23+
},
24+
});
2525

2626
return (
2727
<div className="flex flex-col w-full max-w-md py-24 mx-auto stretch">
@@ -45,7 +45,7 @@ export default function Chat() {
4545
<button
4646
className="px-4 py-2 font-bold text-white bg-blue-500 rounded hover:bg-blue-700"
4747
onClick={() =>
48-
experimental_addToolResult({
48+
addToolResult({
4949
toolCallId,
5050
result: 'Yes, confirmed.',
5151
})
@@ -56,7 +56,7 @@ export default function Chat() {
5656
<button
5757
className="px-4 py-2 font-bold text-white bg-red-500 rounded hover:bg-red-700"
5858
onClick={() =>
59-
experimental_addToolResult({
59+
addToolResult({
6060
toolCallId,
6161
result: 'No, denied',
6262
})

‎packages/core/react/use-chat.ts

+53-37
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
import { useCallback, useEffect, useId, useRef, useState } from 'react';
22
import useSWR, { KeyedMutator } from 'swr';
3-
import { ToolCall as CoreToolCall } from '../core/generate-text/tool-call';
43
import { callChatApi } from '../shared/call-chat-api';
54
import { generateId as generateIdFunc } from '../shared/generate-id';
65
import { processChatStream } from '../shared/process-chat-stream';
@@ -224,6 +223,7 @@ export function useChat({
224223
experimental_onToolCall,
225224
onToolCall,
226225
experimental_maxAutomaticRoundtrips = 0,
226+
maxAutomaticRoundtrips = experimental_maxAutomaticRoundtrips,
227227
streamMode,
228228
onResponse,
229229
onFinish,
@@ -236,6 +236,10 @@ export function useChat({
236236
api?: string | StreamingReactResponseAction;
237237
key?: string;
238238
/**
239+
@deprecated Use `maxAutomaticRoundtrips` instead.
240+
*/
241+
experimental_maxAutomaticRoundtrips?: number;
242+
/**
239243
Maximal number of automatic roundtrips for tool calls.
240244
241245
An automatic tool call roundtrip is a call to the server with the
@@ -247,15 +251,25 @@ case of misconfigured tools.
247251
248252
By default, it's set to 0, which will disable the feature.
249253
*/
250-
experimental_maxAutomaticRoundtrips?: number;
254+
maxAutomaticRoundtrips?: number;
251255
} = {}): UseChatHelpers & {
256+
/**
257+
* @deprecated Use `addToolResult` instead.
258+
*/
252259
experimental_addToolResult: ({
253260
toolCallId,
254261
result,
255262
}: {
256263
toolCallId: string;
257264
result: any;
258265
}) => void;
266+
addToolResult: ({
267+
toolCallId,
268+
result,
269+
}: {
270+
toolCallId: string;
271+
result: any;
272+
}) => void;
259273
} {
260274
// Generate a unique id for the chat if not provided.
261275
const hookId = useId();
@@ -370,12 +384,11 @@ By default, it's set to 0, which will disable the feature.
370384
// ensure there is a last message:
371385
lastMessage != null &&
372386
// check if the feature is enabled:
373-
experimental_maxAutomaticRoundtrips > 0 &&
387+
maxAutomaticRoundtrips > 0 &&
374388
// check that roundtrip is possible:
375389
isAssistantMessageWithCompletedToolCalls(lastMessage) &&
376390
// limit the number of automatic roundtrips:
377-
countTrailingAssistantMessages(messages) <=
378-
experimental_maxAutomaticRoundtrips
391+
countTrailingAssistantMessages(messages) <= maxAutomaticRoundtrips
379392
) {
380393
await triggerRequest({ messages });
381394
}
@@ -396,7 +409,7 @@ By default, it's set to 0, which will disable the feature.
396409
experimental_onFunctionCall,
397410
experimental_onToolCall,
398411
onToolCall,
399-
experimental_maxAutomaticRoundtrips,
412+
maxAutomaticRoundtrips,
400413
messagesRef,
401414
abortControllerRef,
402415
generateId,
@@ -524,6 +537,38 @@ By default, it's set to 0, which will disable the feature.
524537
setInput(e.target.value);
525538
};
526539

540+
const addToolResult = ({
541+
toolCallId,
542+
result,
543+
}: {
544+
toolCallId: string;
545+
result: any;
546+
}) => {
547+
const updatedMessages = messagesRef.current.map((message, index, arr) =>
548+
// update the tool calls in the last assistant message:
549+
index === arr.length - 1 &&
550+
message.role === 'assistant' &&
551+
message.toolInvocations
552+
? {
553+
...message,
554+
toolInvocations: message.toolInvocations.map(toolInvocation =>
555+
toolInvocation.toolCallId === toolCallId
556+
? { ...toolInvocation, result }
557+
: toolInvocation,
558+
),
559+
}
560+
: message,
561+
);
562+
563+
mutate(updatedMessages, false);
564+
565+
// auto-submit when all tool calls in the last assistant message have results:
566+
const lastMessage = updatedMessages[updatedMessages.length - 1];
567+
if (isAssistantMessageWithCompletedToolCalls(lastMessage)) {
568+
triggerRequest({ messages: updatedMessages });
569+
}
570+
};
571+
527572
return {
528573
messages: messages || [],
529574
error,
@@ -537,37 +582,8 @@ By default, it's set to 0, which will disable the feature.
537582
handleSubmit,
538583
isLoading,
539584
data: streamData,
540-
experimental_addToolResult: ({
541-
toolCallId,
542-
result,
543-
}: {
544-
toolCallId: string;
545-
result: any;
546-
}) => {
547-
const updatedMessages = messagesRef.current.map((message, index, arr) =>
548-
// update the tool calls in the last assistant message:
549-
index === arr.length - 1 &&
550-
message.role === 'assistant' &&
551-
message.toolInvocations
552-
? {
553-
...message,
554-
toolInvocations: message.toolInvocations.map(toolInvocation =>
555-
toolInvocation.toolCallId === toolCallId
556-
? { ...toolInvocation, result }
557-
: toolInvocation,
558-
),
559-
}
560-
: message,
561-
);
562-
563-
mutate(updatedMessages, false);
564-
565-
// auto-submit when all tool calls in the last assistant message have results:
566-
const lastMessage = updatedMessages[updatedMessages.length - 1];
567-
if (isAssistantMessageWithCompletedToolCalls(lastMessage)) {
568-
triggerRequest({ messages: updatedMessages });
569-
}
570-
},
585+
addToolResult,
586+
experimental_addToolResult: addToolResult,
571587
};
572588
}
573589

0 commit comments

Comments
 (0)
Please sign in to comment.