Storing Vercel's AI SDK chat messages in a Turso database

Cover image for Storing Vercel's AI SDK chat messages in a Turso database

The Vercel AI SDK gives users a simple yet powerful way to integrate AI into application using various providers and LLMs.

A popular use of AI is in chat based contexts, where the user and LLM can interact in a conversational manner. This can be used for a variety of purposes, such as customer service, chatbots, and more. Vercel makes this particular use case easy with the useChat hook.

Here's how simple Vercel make it to integrate AI into a Next.js app:

// app/chat.tsx
'use client';

import { useChat } from 'ai/react';

export function Chat() {
  const { messages, input, handleInputChange, handleSubmit } = useChat();

  return (
    <>
      {messages.map((message) => (
        <div key={message.id}>
          {message.role === 'user' ? 'User: ' : 'AI: '}
          {message.content}
        </div>
      ))}

      <form onSubmit={handleSubmit}>
        <input name="prompt" value={input} onChange={handleInputChange} />
        <button type="submit">Submit</button>
      </form>
    </>
  );
}

The useChat hook has some sensible defaults, including the location of an API route to handle the AI logic:

// app/api/chat/route.ts
import { openai } from '@ai-sdk/openai';
import { convertToCoreMessages, streamText } from 'ai';

export async function POST(req: Request) {
  const { messages } = await req.json();

  const result = await streamText({
    model: openai('gpt-4-turbo'),
    system: 'You are a helpful assistant.',
    messages: convertToCoreMessages(messages),
  });

  return result.toAIStreamResponse();
}

However, it doesn't provide a way to store the chat messages so when users resume a conversation.

This is where Turso comes in.

#Setting up Turso

If you don't already have a database, you can get one at sqlite.new.

If you used the sqlite.new link, you'll already have the table because we loaded a SQL dump. If you're following along with an existing database, connect to your database using the turso db shell and run the following SQL:

CREATE TABLE messages (
  id TEXT PRIMARY KEY,
  role TEXT,
  content TEXT,
  createdAt TEXT
);

Install the @libsql/client:

npm install @libsql/client

Create a db instance using createClient:

import { createClient } from '@libsql/client';

const db = createClient({
  url: 'libsql://...',
  authToken: '...',
});

#Retrieving chat messages

Let's begin by retrieving the existing chat messages from the database.

Inside your application (e.g. app/page.tsx), import the Message type from the ai package and define a MessageRow interface to represent the database rows:

// app/page.tsx
import type { Message } from 'ai';
import { createClient } from '@libsql/client';

import { Chat } from './chat';

const db = createClient({
  url: 'libsql://...',
  authToken: '...',
});

interface MessageRow {
  id: string;
  role: string;
  content: string;
  createdAt: string;
}

Now we can query (also inside app/page.tsx) the database and transform the rows into the Message type:

// app/page.tsx
export default async function Page() {
  const result = await db.execute('SELECT * FROM messages');
  const rows = result.rows as unknown as Array<MessageRow>;

  const messages = rows.map(({ id, role, content, createdAt }) => ({
    id,
    role: role as Message['role'],
    content,
    createdAt: new Date(createdAt),
  }));

  return <Chat initialMessages={messages} />;
}

We're currently passing initialMessages to the <Chat /> component we defined earlier. Let's update it to accept this new prop:

// app/chat.tsx
'use client';

import type { Message } from 'ai';
import { useChat } from 'ai/react';

export function Chat({ initialMessages = [] }: { initialMessages: Message[] }) {
  // ...
}

Now pass initialMessages to the useChat hook:

// app/chat.tsx

export function Chat({ initialMessages = [] }: { initialMessages: Message[] }) {
  const { messages, input, handleInputChange, handleSubmit } = useChat({
    initialMessages,
  });

  return (
    <>
      {messages.map((message) => (
        <div key={message.id}>
          {message.role === 'user' ? 'User: ' : 'AI: '}
          {message.content}
        </div>
      ))}

      <form onSubmit={handleSubmit}>
        <input name="prompt" value={input} onChange={handleInputChange} />
        <button type="submit">Submit</button>
      </form>
    </>
  );
}

#Storing user messages

We're going to a cheat a little bit for the purposes of this tutorial, and assume the last message sent to the /api/chat route is not already in the database, and is what the user typed.

// app/api/chat/route.ts
export async function POST(req: Request) {
  const { messages } = await req.json();

  const lastUserMessage = messages[messages.length - 1];

  const result = await streamText({
    model: openai('gpt-4-turbo'),
    messages,
  });

  return new StreamingTextResponse(result.toAIStream());
}

The lastUserMessage looks something like:

{ "role": "user", "content": "Why is Next.js so good?" }

If we refer back to the MessageRow type we had earlier, you'll notice two additional fields id and createdAt.

Let's make sure we create values for those:

// app/api/chat/route.ts
const id = crypto.randomUUID();
const createdAt = new Date().toISOString();

Now we can insert the message into the database:

// app/api/chat/route.ts
const id = crypto.randomUUID();
const createdAt = new Date().toISOString();

await db.execute({
  sql: 'INSERT INTO messages (id, role, content, createdAt) VALUES (?, ?, ?, ?)',
  args: [id, lastUserMessage.role, lastUserMessage.content, createdAt],
});

We're now successfully storing messages in the database that were sent by the user, but if we refresh the page, we won't see the message from OpenAI.

#Storing assistant messages

To store the response from OpenAI, we can pass an onFinish function to streamText():

const result = await streamText({
  model: openai('gpt-4-turbo'),
  messages,
  onFinish: async ({ text }) => {
    const id = crypto.randomUUID();
    const createdAt = new Date().toISOString();

    await db.execute({
      sql: 'INSERT INTO messages (id, role, content, createdAt) VALUES (?, ?, ?, ?)',
      args: [id, 'assistant', text, createdAt],
    });
  },
});

Now whenever we send a message, and the AI responds, both messages will be saved in the database.

If you refresh the page, or close it and return later, you will have the full message history in context.

#Going further

This tutorial covered how to use the existing conventions of useChat to add content to your database, however, in a real-world application, you may want to consider using a server action and process storing the message to the database in the background.

We also used a single table for storing messages for one chat. You can expand on this example by creating a table for "chats" or "sessions" and associating each message to that with a foreign key.

Turso gives you 500 databases for free to get started, so you can experiment with different schemas and data structures — as well as a database per user with multi-db schemas, or per chat!

scarf