diff --git a/frontend/app/api/connection-details/route.ts b/frontend/app/api/connection-details/route.ts new file mode 100644 index 0000000..d46c1c2 --- /dev/null +++ b/frontend/app/api/connection-details/route.ts @@ -0,0 +1,68 @@ +import { + AccessToken, + AccessTokenOptions, + VideoGrant, +} from "livekit-server-sdk"; +import { NextResponse } from "next/server"; + +const API_KEY = process.env.LIVEKIT_API_KEY; +const API_SECRET = process.env.LIVEKIT_API_SECRET; +const LIVEKIT_URL = process.env.LIVEKIT_URL; + +export type ConnectionDetails = { + serverUrl: string; + roomName: string; + participantName: string; + participantToken: string; +}; + +export async function GET() { + try { + // Generate participant token + const participantIdentity = `voice_assistant_user_${Math.round( + Math.random() * 10_000 + )}`; + const participantToken = await createParticipantToken( + { + identity: participantIdentity, + }, + "roomName" + ); + + if (LIVEKIT_URL === undefined) { + throw new Error("LIVEKIT_URL is not defined"); + } + + // Return connection details + const data: ConnectionDetails = { + serverUrl: LIVEKIT_URL, + roomName: "voice_assistant_room", + participantToken: participantToken, + participantName: participantIdentity, + }; + return NextResponse.json(data); + } catch (error) { + if (error instanceof Error) { + console.error(error); + + return new NextResponse(error.message, { status: 500 }); + } + } +} + +function createParticipantToken( + userInfo: AccessTokenOptions, + roomName: string +) { + const at = new AccessToken(API_KEY, API_SECRET, userInfo); + at.ttl = "5m"; + const grant: VideoGrant = { + room: roomName, + roomJoin: true, + canPublish: true, + canPublishData: true, + canSubscribe: true, + }; + at.addGrant(grant); + return at.toJwt(); +} diff --git a/frontend/app/page.tsx b/frontend/app/page.tsx index d3e3b95..d13cfbd 100644 --- a/frontend/app/page.tsx +++ b/frontend/app/page.tsx @@ -11,7 +11,7 @@ import { } from "@livekit/components-react"; import { useCallback, useState } from "react"; import { MediaDeviceFailure } from "livekit-client"; -import type { ConnectionDetails } from "./connection-details/route"; +import type { ConnectionDetails } from "./api/connection-details/route"; export default function Page() { const [connectionDetails, updateConnectionDetails] = useState< @@ -20,7 +20,8 @@ export default function Page() { const onConnectButtonClicked = useCallback(async () => { const url = new URL( - process.env.NEXT_PUBLIC_CONN_DETAILS_ENDPOINT ?? "/connection-details", + process.env.NEXT_PUBLIC_CONN_DETAILS_ENDPOINT ?? + "/api/connection-details", window.location.origin ); const response = await fetch(url.toString());