AI SDK RSCHandling Loading State

Handling Loading State

Given that responses from language models can often take a while to complete, it's crucial to be able to show loading state to users. This provides visual feedback that the system is working on their request and helps maintain a positive user experience.

There are three approaches you can take to handle loading state with the AI SDK RSC:

  • Managing loading state similar to how you would in a traditional Next.js application. This involves setting a loading state variable in the client and updating it when the response is received.
  • Streaming loading state from the server to the client. This approach allows you to track loading state on a more granular level and provide more detailed feedback to the user.
  • Streaming loading component from the server to the client. This approach allows you to stream a React Server Component to the client while awaiting the model's response.

Handling Loading State on the Client

Client

Let's create a simple Next.js page that will call the generateResponse function when the form is submitted. The function will take in the user's prompt (input) and then generate a response (response). To handle the loading state, use the loading state variable. When the form is submitted, set loading to true, and when the response is received, set it back to false. While the response is being streamed, the input field will be disabled.

app/page.tsx
'use client';
import { useState } from 'react';
import { generateResponse } from './actions';
import { readStreamableValue } from 'ai/rsc';
// Force the page to be dynamic and allow streaming responses up to 30 seconds
export const maxDuration = 30;
export default function Home() {
const [input, setInput] = useState<string>('');
const [generation, setGeneration] = useState<string>('');
const [loading, setLoading] = useState<boolean>(false);
return (
<div>
<div>{generation}</div>
<form
onSubmit={async e => {
e.preventDefault();
setLoading(true);
const response = await generateResponse(input);
let textContent = '';
for await (const delta of readStreamableValue(response)) {
textContent = `${textContent}${delta}`;
setGeneration(textContent);
}
setInput('');
setLoading(false);
}}
>
<input
type="text"
value={input}
disabled={loading}
className="disabled:opacity-50"
onChange={event => {
setInput(event.target.value);
}}
/>
<button>Send Message</button>
</form>
</div>
);
}

Server

Now let's implement the generateResponse function. Use the streamText function to generate a response to the input.

app/actions.ts
'use server';
import { streamText } from 'ai';
import { openai } from '@ai-sdk/openai';
import { createStreamableValue } from 'ai/rsc';
export async function generateResponse(prompt: string) {
const stream = createStreamableValue();
(async () => {
const { textStream } = await streamText({
model: openai('gpt-4o'),
prompt,
});
for await (const text of textStream) {
stream.update(text);
}
stream.done();
})();
return stream.value;
}

Streaming Loading State from the Server

If you are looking to track loading state on a more granular level, you can create a new streamable value to store a custom variable and then read this on the frontend. Let's update the example to create a new streamable value for tracking loading state:

Server

app/actions.ts
'use server';
import { streamText } from 'ai';
import { openai } from '@ai-sdk/openai';
import { createStreamableValue } from 'ai/rsc';
export async function generateResponse(prompt: string) {
const stream = createStreamableValue();
const loadingState = createStreamableValue({ loading: true });
(async () => {
const { textStream } = await streamText({
model: openai('gpt-4o'),
prompt,
});
for await (const text of textStream) {
stream.update(text);
}
stream.done();
loadingState.done({ loading: false });
})();
return { response: stream.value, loadingState: loadingState.value };
}

Client

app/page.tsx
'use client';
import { useState } from 'react';
import { generateResponse } from './actions';
import { readStreamableValue } from 'ai/rsc';
// Force the page to be dynamic and allow streaming responses up to 30 seconds
export const maxDuration = 30;
export default function Home() {
const [input, setInput] = useState<string>('');
const [generation, setGeneration] = useState<string>('');
const [loading, setLoading] = useState<boolean>(false);
return (
<div>
<div>{generation}</div>
<form
onSubmit={async e => {
e.preventDefault();
setLoading(true);
const { response, loadingState } = await generateResponse(input);
let textContent = '';
for await (const responseDelta of readStreamableValue(response)) {
textContent = `${textContent}${responseDelta}`;
setGeneration(textContent);
}
for await (const loadingDelta of readStreamableValue(loadingState)) {
if (loadingDelta) {
setLoading(loadingDelta.loading);
}
}
setInput('');
setLoading(false);
}}
>
<input
type="text"
value={input}
disabled={loading}
className="disabled:opacity-50"
onChange={event => {
setInput(event.target.value);
}}
/>
<button>Send Message</button>
</form>
</div>
);
}

This allows you to provide more detailed feedback about the generation process to your users.

Streaming Loading Components with streamUI

If you are using the streamUI function, you can stream the loading state to the client in the form of a React component. streamUI supports the usage of JavaScript generator functions , which allow you to yield some value (in this case a React component) while some other blocking work completes.

Server

'use server';
import { openai } from '@ai-sdk/openai';
import { streamUI } from 'ai/rsc';
export async function generateResponse(prompt: string) {
const result = await streamUI({
model: openai('gpt-4o'),
prompt,
text: async function* ({ content }) {
yield <div>loading...</div>;
return <div>{content}</div>;
},
});
return result.value;
}

Remember to update the file from .ts to .tsx because you are defining a React component in the streamUI function.

Client

'use client';
import { useState } from 'react';
import { generateResponse } from './actions';
import { readStreamableValue } from 'ai/rsc';
// Force the page to be dynamic and allow streaming responses up to 30 seconds
export const maxDuration = 30;
export default function Home() {
const [input, setInput] = useState<string>('');
const [generation, setGeneration] = useState<React.ReactNode>();
return (
<div>
<div>{generation}</div>
<form
onSubmit={async e => {
e.preventDefault();
const result = await generateResponse(input);
setGeneration(result);
setInput('');
}}
>
<input
type="text"
value={input}
onChange={event => {
setInput(event.target.value);
}}
/>
<button>Send Message</button>
</form>
</div>
);
}