Client Code
Tidying up your Next.js project
The default project template includes additional files that we don’t need for this tutorial.
Let’s delete the following files:
public/next.svg
public/vercel.svg
We’ll remove the placeholder content from /app/page.tsx
:
It is important to note the "use-client";
directive at the very top of the file. This instructs Next.js to only render this code on the client side (in the browser). This is important because we will be using the RTVIClient
, which is a browser-based client.
Creating the RTVIClient
The Pipecat JS SDK package includes an RTVIClient
class that we can use to connect to our bot.
Let’s create a new instance of the RTVIClient
in our Home
component and provide it with some initial configuration:
baseUrl
The baseUrl
option is the URL that the RTVIClient
will use to request an authentication bundle (a Daily room URL and token) from the Daily Bots API, and start a bot in the process.
In this case, we are using the /api
route we just created (/app/api/route.ts).
services
The services map defines which providers you’d like to use for each function of your bot, in this case the speech-to-text (STT), LLM and text-to-speech (TTS) services. The service map can either be specified on your server-side route or passed from the client-side to the server-side route. If you’re doing the latter, you can define it within the params.requestData
object, which will make it available in the request body on the server-side.
In the above example, we are using Deepgram for STT, Anthropic as the LLM and Cartesia for TTS.
You can read more about which services we currently support, as well as how to provide your own API keys here.
config
The config
array contains the configuration for each service and the options you’d like to pass to them.
In the above example, we are setting the voice for the TTS service to 79a125e8-cd45-4c13-8a67-188112f4dd22
and the model for the LLM service to meta-llama/Meta-Llama-3.1-70B-Instruct-Turbo
.
The service string must match that of a service you have defined in your services
map.
Why is config an array?
Initially, it might seem a little strange to use an array vs. a map for the config object.
The RTVI standard is designed with flexibility in mind, and passing an array allows you to define the order in which you want your config options to run.
Daily Bots are built on Pipecat, an open-source pipeline orchestration framework that allows you to chain together multiple services.
Pipecat works by dispatching ‘frames’ in a specific order, and the order in which you define options in the config
array will determine the sequence in which they are dispatched.
This allows for a high degree of flexibility in how you structure your bot’s logic.
Remember: the order in which you define your service and config options in the array is important!
We also define { name: "run_on_config", value: true },
This will instruct the bot to run the LLM service as soon as it connects. If you’d prefer to start the conversation, you can remove this line (or set it to false
).
To emphasize the importance of config ordering, let’s consider run_on_config
as an example.
If we were to move the run_on_config
option above initial_messages
in our config array, the Pipecat bot would consume this and run the LLM service before configuring the initial messages. This would result in a garbled response from the LLM, as it has no context messages to work with.
Adding an audio component
We need a way to hear our bot once we have connected. The @pipecat-ai/client-react
package includes a handy RTVIClientAudio
component that will subscribe to the bot’s incoming audio track.
In order to use this, we’ll need to wrap our project inside a RTVIClientProvider
context.
Using this context means we can now use the hooks and components provided by @pipecat-ai/client-react
to interact with our bot.
Starting the session
Now that we have our RTVIClient
instance set up, we can start the session by calling the connect
method.
Let’s create a new App.tsx
component in app/
and add the following code:
And update our page.tsx
to use this component, making sure to wrap it in the RTVIClientProvider
:
Calling the connect() method
We’ll add a button to connect to the Daily Bot (and disable it once we’re connected):
Click the button, and you should hear your bot say hello!
Please ensure you accept your browser’s permissions to use your microphone.
Why is connect() async?
The connect
method can be invoked either asynchronously or synchronously. If called with await
, the returned promise will resolve once the bot enters into a ‘ready’ state (indicating that the bot is connected and ready to receive input).
Calling it asynchronously also allows us to try / catch
any errors that may occur during the connection process.
If you’d rather not use asynchronous code, you can invoke connect
without await
and use the various events and callbacks provided by RTVI.
Wrapping up
Let’s add a bit more functionality to our app. We’ll add an onscreen transcription of the bot’s responses, and disable the start button once we’re connected.