diff --git a/bun.lock b/bun.lock index 39887102f..2d762badc 100644 --- a/bun.lock +++ b/bun.lock @@ -15,6 +15,7 @@ "name": "@opencode/cloud-app", "dependencies": { "@ibm/plex": "6.4.1", + "@openauthjs/openauth": "0.0.0-20250322224806", "@solidjs/meta": "^0.29.4", "@solidjs/router": "^0.15.0", "@solidjs/start": "^1.1.0", diff --git a/cloud/app/package.json b/cloud/app/package.json index 61b3522a7..ba6ab6427 100644 --- a/cloud/app/package.json +++ b/cloud/app/package.json @@ -9,6 +9,7 @@ }, "dependencies": { "@ibm/plex": "6.4.1", + "@openauthjs/openauth": "0.0.0-20250322224806", "@solidjs/meta": "^0.29.4", "@solidjs/router": "^0.15.0", "@solidjs/start": "^1.1.0", diff --git a/cloud/app/src/context/auth.tsx b/cloud/app/src/context/auth.tsx new file mode 100644 index 000000000..bec949568 --- /dev/null +++ b/cloud/app/src/context/auth.tsx @@ -0,0 +1,28 @@ +import { useSession } from "vinxi/http" +import { createClient } from "@openauthjs/openauth/client" + +export const AuthClient = createClient({ + clientID: "app", + issuer: "https://auth.dev.opencode.ai", +}) + +export interface AuthSession { + account: Record + current?: string +} + +export function useAuthSession() { + "use server" + + return useSession({ + password: "0".repeat(32), + name: "auth" + }) +} + + +export function AuthProvider() { +} diff --git a/cloud/app/src/routes/auth/authorize.ts b/cloud/app/src/routes/auth/authorize.ts new file mode 100644 index 000000000..166466ef8 --- /dev/null +++ b/cloud/app/src/routes/auth/authorize.ts @@ -0,0 +1,7 @@ +import type { APIEvent } from "@solidjs/start/server" +import { AuthClient } from "~/context/auth" + +export async function GET(input: APIEvent) { + const result = await AuthClient.authorize(new URL("./callback", input.request.url).toString(), "code") + return Response.redirect(result.url, 302) +} diff --git a/cloud/app/src/routes/auth/callback.ts b/cloud/app/src/routes/auth/callback.ts new file mode 100644 index 000000000..a561c70d0 --- /dev/null +++ b/cloud/app/src/routes/auth/callback.ts @@ -0,0 +1,36 @@ +import type { APIEvent } from "@solidjs/start/server" +import { AuthClient, useAuthSession } from "~/context/auth" + +export async function GET(input: APIEvent) { + const url = new URL(input.request.url) + const code = url.searchParams.get("code") + if (!code) throw new Error("No code found") + const redirectURI = `${url.origin}${url.pathname}` + console.log({ + redirectURI, + code, + }) + const result = await AuthClient.exchange(code, `${url.origin}${url.pathname}`) + if (result.err) { + throw new Error(result.err.message) + } + const decoded = AuthClient.decode(result.tokens.access, {} as any) + if (decoded.err) throw new Error(decoded.err.message) + const session = await useAuthSession() + const id = decoded.subject.properties.accountID + await session.update((value) => { + return { + ...value, + account: { + [id]: { + id, + email: decoded.subject.properties.email, + }, + }, + current: id, + } + }) + return { + result, + } +} diff --git a/cloud/app/src/routes/index.tsx b/cloud/app/src/routes/index.tsx index 7c7eee13d..da4e23364 100644 --- a/cloud/app/src/routes/index.tsx +++ b/cloud/app/src/routes/index.tsx @@ -51,7 +51,7 @@ export default function Home() { Get Started
-