update cf context to studio
This commit is contained in:
@@ -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}`)
|
||||||
|
|||||||
@@ -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(
|
||||||
|
|||||||
@@ -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) }
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -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">
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
Reference in New Issue
Block a user