Installation

To get started with Next.js and bknd you can either install the package manually, and follow the descriptions below, or use the CLI starter:

Create a new Next.js CLI starter project by running the following command:

npx bknd create -i nextjs

Serve the API

Create a helper file to instantiate the bknd instance and retrieve the API:

src/bknd.ts
import { type NextjsBkndConfig, getApp as getBkndApp } from "bknd/adapter/nextjs";
import { headers } from "next/headers";

export const config = {
   connection: {
      url: "file:data.db"
   },
} as const satisfies NextjsBkndConfig;

export async function getApp() {
   return await getBkndApp(config);
}

export async function getApi(opts?: { verify?: boolean }) {
   const app = await getApp();
   if (opts?.verify) {
      const api = app.getApi({ headers: await headers() });
      await api.verifyAuth();
      return api;
   }

   return app.getApi();
}

For more information about the connection object, refer to the Database guide.

Now to expose the API, create a catch-all route file at src/api/[[...bknd]]/route.ts:

src/api/[[...bknd]]/route.ts
import { config } from "@/bknd";
import { serve } from "bknd/adapter/nextjs";

// optionally, you can set the runtime to edge for better performance
export const runtime = "edge";

const handler = serve({
   ...config,
   cleanRequest: {
      // depending on what name you used for the catch-all route,
      // you need to change this to clean it from the request.
      searchParams: ["bknd"],
   },
});

export const GET = handler;
export const POST = handler;
export const PUT = handler;
export const PATCH = handler;
export const DELETE = handler;

Enabling the Admin UI

Create a page at admin/[[...admin]]/page.tsx:

admin/[[...admin]]/page.tsx
import { Admin } from "bknd/ui";
import { getApi } from "@/bknd";
import "bknd/dist/styles.css";

export default async function AdminPage() {
   // make sure to verify auth using headers
   const api = await getApi({ verify: true });

   return (
      <Admin
         withProvider={{ user: api.getUser() }}
         config={{
            basepath: "/admin",
            logo_return_path: "/../",
            color_scheme: "system",
         }}
      />
   );
}

Example usage of the API

You can use the getApi helper function we’ve already set up to fetch and mutate in static pages and server components:

app/page.tsx
import { getApi } from "@/bknd";

export default async function Home() {
   const api = await getApi();
   const { data: todos } = await api.data.readMany("todos", { limit: 5 });

   return <ul>
      {todos.map((todo) => (
         <li key={String(todo.id)}>{todo.title}</li>
      ))}
   </ul>
}