update cf context to studio

This commit is contained in:
2025-12-18 00:15:18 -03:00
parent 8c7c3034d4
commit 3857620f89
7 changed files with 52 additions and 40 deletions

View File

@@ -1,7 +1,7 @@
import type { Route } from './+types' import type { Route } from './+types'
import type { User } from '@repo/auth/auth' import type { User } from '@repo/auth/auth'
import { userContext } from '@repo/auth/context' import { userContext, cloudflareContext } from '@repo/auth/context'
export const loader = proxy export const loader = proxy
export const action = proxy export const action = proxy
@@ -11,8 +11,9 @@ async function proxy({
context context
}: Route.ActionArgs): Promise<Response> { }: Route.ActionArgs): Promise<Response> {
const pathname = new URL(request.url).pathname.replace(/^\/api\//, '') const pathname = new URL(request.url).pathname.replace(/^\/api\//, '')
const user = context.get(userContext) as User const cloudflare = context.get(cloudflareContext)
const url = new URL(pathname, context.cloudflare.env.API_URL) const user = context.get(userContext)!
const url = new URL(pathname, cloudflare.env.API_URL)
const headers = new Headers(request.headers) const headers = new Headers(request.headers)
headers.set('Authorization', `Bearer ${user.accessToken}`) headers.set('Authorization', `Bearer ${user.accessToken}`)

View File

@@ -1,26 +1,34 @@
import type { Route } from './+types' import type { Route } from './+types/login'
import { redirect } from 'react-router' import { redirect } from 'react-router'
import { createAuth, type User } from '@repo/auth/auth' import { createAuth, type User } from '@repo/auth/auth'
import { requestIdContext, cloudflareContext } from '@repo/auth/context'
import { createSessionStorage } from '@repo/auth/session' import { createSessionStorage } from '@repo/auth/session'
export async function loader({ request, context }: Route.ActionArgs) { export async function loader({ request, context }: Route.LoaderArgs) {
const sessionStorage = createSessionStorage(context.cloudflare.env) const url = new URL(request.url)
const cloudflare = context.get(cloudflareContext)
const sessionStorage = createSessionStorage(cloudflare.env)
const session = await sessionStorage.getSession(request.headers.get('cookie')) const session = await sessionStorage.getSession(request.headers.get('cookie'))
const returnTo = session.has('returnTo') ? session.get('returnTo') : '/' const requestId = context.get(requestIdContext)
const user = session.get('user') as User | null const user = session.get('user')
const returnTo = (session.get('returnTo') as string | undefined) ?? '/'
if (user) { if (user) {
return redirect(returnTo) return redirect(returnTo)
} }
try { try {
const authenticator = createAuth(context.cloudflare.env) const authenticator = createAuth(cloudflare.env, `${url.origin}/login`)
const user = await authenticator.authenticate('oidc', request) const authenticatedUser = (await authenticator.authenticate(
session.set('user', user) 'oidc',
request
)) as User
session.set('user', authenticatedUser)
console.log(`[${requestId}] Redirecting the user to ${returnTo}`)
console.log(`Redirecting the user to ${returnTo}`)
// Redirect to the home page after successful login // Redirect to the home page after successful login
return redirect(returnTo, { return redirect(returnTo, {
headers: { headers: {
@@ -28,7 +36,7 @@ export async function loader({ request, context }: Route.ActionArgs) {
} }
}) })
} catch (error) { } catch (error) {
console.error(error) console.error(`[${requestId}]`, error)
if (error instanceof Error) { if (error instanceof Error) {
return Response.json( return Response.json(

View File

@@ -1,23 +1,24 @@
import type { Route } from './+types' import type { Route } from './+types/logout'
import { createAuth, type User } from '@repo/auth/auth'
import { createSessionStorage } from '@repo/auth/session'
import { redirect } from 'react-router' import { redirect } from 'react-router'
import type { OAuth2Strategy } from 'remix-auth-oauth2' import type { OAuth2Strategy } from 'remix-auth-oauth2'
import { createAuth, type User } from '@repo/auth/auth'
import { createSessionStorage } from '@repo/auth/session'
import { cloudflareContext } from '@repo/auth/context'
export async function loader({ request, context }: Route.LoaderArgs) { export async function loader({ request, context }: Route.LoaderArgs) {
const authenticator = createAuth(context.cloudflare.env) const cloudflare = context.get(cloudflareContext)
const sessionStorage = createSessionStorage(context.cloudflare.env) const authenticator = createAuth(cloudflare.env)
const sessionStorage = createSessionStorage(cloudflare.env)
const session = await sessionStorage.getSession(request.headers.get('cookie')) const session = await sessionStorage.getSession(request.headers.get('cookie'))
const user = session.get('user') as User const user = session.get('user')
const strategy = authenticator.get<OAuth2Strategy<User>>('oidc') const strategy = authenticator.get<OAuth2Strategy<User>>('oidc')
if (user?.accessToken && strategy) { if (user?.accessToken && strategy) {
await strategy.revokeToken(user.accessToken) await strategy.revokeToken(user.accessToken)
} }
console.log(await sessionStorage.destroySession(session))
return redirect('/login', { return redirect('/login', {
headers: { 'Set-Cookie': await sessionStorage.destroySession(session) } headers: { 'Set-Cookie': await sessionStorage.destroySession(session) }
}) })

View File

@@ -131,7 +131,9 @@ export async function action({ params, request, context }: Route.ActionArgs) {
return { ok: r.status === 200 } return { ok: r.status === 200 }
} }
export default function Component({ loaderData: { data } }) { export default function Component({
loaderData: { data }
}: Route.ComponentProps) {
return ( return (
<Suspense fallback={<Skeleton />}> <Suspense fallback={<Skeleton />}>
<div className="space-y-4"> <div className="space-y-4">

View File

@@ -11,6 +11,7 @@ import {
import { Suspense, useMemo } from 'react' import { Suspense, useMemo } from 'react'
import { Await, NavLink, useSearchParams } from 'react-router' import { Await, NavLink, useSearchParams } from 'react-router'
import { cloudflareContext } from '@repo/auth/context'
import { createSearch } from '@repo/util/meili' import { createSearch } from '@repo/util/meili'
import { SearchForm } from '@repo/ui/components/search-form' import { SearchForm } from '@repo/ui/components/search-form'
import { Skeleton } from '@repo/ui/components/skeleton' import { Skeleton } from '@repo/ui/components/skeleton'
@@ -44,11 +45,12 @@ export function meta({}: Route.MetaArgs) {
} }
export const loader = async ({ context }: Route.ActionArgs) => { export const loader = async ({ context }: Route.ActionArgs) => {
const cloudflare = context.get(cloudflareContext)
const courses = createSearch({ const courses = createSearch({
index: 'saladeaula_courses', index: 'saladeaula_courses',
sort: ['created_at:desc'], sort: ['created_at:desc'],
hitsPerPage: 100, hitsPerPage: 100,
env: context.cloudflare.env env: cloudflare.env
}) })
return { return {
@@ -56,7 +58,9 @@ export const loader = async ({ context }: Route.ActionArgs) => {
} }
} }
export default function Component({ loaderData: { data } }) { export default function Component({
loaderData: { data }
}: Route.ComponentProps) {
const [searchParams, setSearchParams] = useSearchParams() const [searchParams, setSearchParams] = useSearchParams()
const term = searchParams.get('term') as string const term = searchParams.get('term') as string
@@ -79,13 +83,13 @@ export default function Component({ loaderData: { data } }) {
} }
defaultValue={term} defaultValue={term}
onChange={(term) => { onChange={(term) => {
setSearchParams({ term }) setSearchParams({ term: String(term) })
}} }}
/> />
</div> </div>
<Await resolve={data}> <Await resolve={data}>
{({ hits = [] }) => <List term={term} hits={hits} />} {({ hits }) => <List term={term} hits={hits as Course[]} />}
</Await> </Await>
</div> </div>
</Suspense> </Suspense>

View File

@@ -1,4 +1,4 @@
import type { Route } from './+types' import type { Route } from './+types/layout'
import { Link, Outlet } from 'react-router' import { Link, Outlet } from 'react-router'
@@ -11,7 +11,7 @@ export const middleware: Route.MiddlewareFunction[] = [authMiddleware]
export async function loader({ context }: Route.ActionArgs) { export async function loader({ context }: Route.ActionArgs) {
const user = context.get(userContext) const user = context.get(userContext)
return Response.json({ user }) return { user }
} }
export default function Component({ loaderData }: Route.ComponentProps) { export default function Component({ loaderData }: Route.ComponentProps) {

View File

@@ -1,12 +1,9 @@
import { createRequestHandler, RouterContextProvider } from 'react-router' import { createRequestHandler, RouterContextProvider } from 'react-router'
import { cloudflareContext } from '@repo/auth/context'
declare module 'react-router' { declare module '@repo/auth/context' {
export interface AppLoadContext { interface CloudflareEnv extends Env {}
cloudflare: { interface CloudflareCtx extends ExecutionContext {}
env: Env
ctx: ExecutionContext
}
}
} }
const requestHandler = createRequestHandler( const requestHandler = createRequestHandler(
@@ -16,11 +13,10 @@ const requestHandler = createRequestHandler(
export default { export default {
async fetch(request, env, ctx) { async fetch(request, env, ctx) {
const context = new RouterContextProvider() const context = new RouterContextProvider(
new Map([[cloudflareContext, { env, ctx }]])
return requestHandler(
request,
Object.assign(context, { cloudflare: { env, ctx } })
) )
return requestHandler(request, context)
} }
} satisfies ExportedHandler<Env> } satisfies ExportedHandler<Env>