add cloudflare context to admin

This commit is contained in:
2025-12-18 00:04:48 -03:00
parent be1ae656d3
commit 8c7c3034d4
16 changed files with 67 additions and 50 deletions

View File

@@ -22,6 +22,7 @@ import {
} from '@repo/ui/components/ui/empty' } from '@repo/ui/components/ui/empty'
import { Kbd } from '@repo/ui/components/ui/kbd' import { Kbd } from '@repo/ui/components/ui/kbd'
import { cn } from '@repo/ui/lib/utils' import { cn } from '@repo/ui/lib/utils'
import { cloudflareContext } from '@repo/auth/context'
import { createSearch } from '@repo/util/meili' import { createSearch } from '@repo/util/meili'
import { request as req } from '@repo/util/request' import { request as req } from '@repo/util/request'
@@ -49,12 +50,13 @@ export function meta({}: Route.MetaArgs) {
} }
export async function loader({ context, request, params }: Route.LoaderArgs) { export async function loader({ context, request, params }: Route.LoaderArgs) {
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'],
filter: 'unlisted NOT EXISTS', filter: 'unlisted NOT EXISTS',
hitsPerPage: 100, hitsPerPage: 100,
env: context.cloudflare.env env: cloudflare.env
}) })
const customPricing = req({ const customPricing = req({

View File

@@ -12,9 +12,8 @@ import {
import { request as req } from '@repo/util/request' import { request as req } from '@repo/util/request'
export async function loader({ params, request, context }: Route.LoaderArgs) { export async function loader({ params, request, context }: Route.LoaderArgs) {
const { id } = params
const r = await req({ const r = await req({
url: `/enrollments/${id}`, url: `/enrollments/${params.id}`,
request, request,
context context
}) })

View File

@@ -15,6 +15,7 @@ import { Button } from '@repo/ui/components/ui/button'
import { ExportMenu } from '@repo/ui/components/export-menu' import { ExportMenu } from '@repo/ui/components/export-menu'
import { Kbd } from '@repo/ui/components/ui/kbd' import { Kbd } from '@repo/ui/components/ui/kbd'
import { createSearch } from '@repo/util/meili' import { createSearch } from '@repo/util/meili'
import { cloudflareContext } from '@repo/auth/context'
import { headers, sortings, statuses } from '@repo/ui/routes/enrollments/data' import { headers, sortings, statuses } from '@repo/ui/routes/enrollments/data'
import { columns, type Enrollment } from './columns' import { columns, type Enrollment } from './columns'
@@ -24,6 +25,7 @@ export function meta({}: Route.MetaArgs) {
} }
export async function loader({ params, context, request }: Route.LoaderArgs) { export async function loader({ params, context, request }: Route.LoaderArgs) {
const cloudflare = context.get(cloudflareContext)
const { searchParams } = new URL(request.url) const { searchParams } = new URL(request.url)
const { orgid } = params const { orgid } = params
const query = searchParams.get('q') || '' const query = searchParams.get('q') || ''
@@ -52,7 +54,7 @@ export async function loader({ params, context, request }: Route.LoaderArgs) {
query, query,
page, page,
hitsPerPage, hitsPerPage,
env: context.cloudflare.env env: cloudflare.env
}) })
return { return {
@@ -85,7 +87,7 @@ export default function Route({ loaderData: { data } }: Route.ComponentProps) {
</div> </div>
<Await resolve={data}> <Await resolve={data}>
{({ hits, page, hitsPerPage, totalHits }) => ( {({ hits, page = 1, hitsPerPage, totalHits }) => (
<DataTable <DataTable
sort={[{ id: 'created_at', desc: true }]} sort={[{ id: 'created_at', desc: true }]}
columns={columns} columns={columns}

View File

@@ -66,6 +66,7 @@ import { createSearch } from '@repo/util/meili'
import { cn } from '@repo/ui/lib/utils' import { cn } from '@repo/ui/lib/utils'
import { HttpMethod, request as req } from '@repo/util/request' import { HttpMethod, request as req } from '@repo/util/request'
import { useIsMobile } from '@repo/ui/hooks/use-mobile' import { useIsMobile } from '@repo/ui/hooks/use-mobile'
import { cloudflareContext } from '@repo/auth/context'
import { formSchema, type Schema, MAX_ITEMS } from './data' import { formSchema, type Schema, MAX_ITEMS } from './data'
import { AsyncCombobox } from './async-combobox' import { AsyncCombobox } from './async-combobox'
@@ -76,12 +77,13 @@ export function meta({}: Route.MetaArgs) {
} }
export async function loader({ context }: Route.LoaderArgs) { export async function loader({ context }: Route.LoaderArgs) {
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'],
filter: 'unlisted NOT EXISTS', filter: 'unlisted NOT EXISTS',
hitsPerPage: 100, hitsPerPage: 100,
env: context.cloudflare.env env: cloudflare.env
}) })
return { courses } return { courses }

View File

@@ -7,6 +7,7 @@ import { MeiliSearchFilterBuilder } from 'meilisearch-helper'
import { DataTable } from '@repo/ui/components/data-table' import { DataTable } from '@repo/ui/components/data-table'
import { Skeleton } from '@repo/ui/components/skeleton' import { Skeleton } from '@repo/ui/components/skeleton'
import { createSearch } from '@repo/util/meili' import { createSearch } from '@repo/util/meili'
import { cloudflareContext } from '@repo/auth/context'
import { columns, type Order } from './columns' import { columns, type Order } from './columns'
@@ -15,14 +16,18 @@ export function meta({}: Route.MetaArgs) {
} }
export async function loader({ params, context, request }: Route.LoaderArgs) { export async function loader({ params, context, request }: Route.LoaderArgs) {
const cloudflare = context.get(cloudflareContext)
const { searchParams } = new URL(request.url) const { searchParams } = new URL(request.url)
const { orgid } = params
const query = searchParams.get('q') || '' const query = searchParams.get('q') || ''
const sort = searchParams.get('sort') || 'create_date:desc' const sort = searchParams.get('sort') || 'create_date:desc'
const page = Number(searchParams.get('p')) + 1 const page = Number(searchParams.get('p')) + 1
const hitsPerPage = Number(searchParams.get('perPage')) || 25 const hitsPerPage = Number(searchParams.get('perPage')) || 25
let builder = new MeiliSearchFilterBuilder().where('tenant_id', '=', orgid) let builder = new MeiliSearchFilterBuilder().where(
'tenant_id',
'=',
params.orgid
)
const orders = createSearch({ const orders = createSearch({
index: 'betaeducacao-prod-orders', index: 'betaeducacao-prod-orders',
@@ -31,7 +36,7 @@ export async function loader({ params, context, request }: Route.LoaderArgs) {
query, query,
page, page,
hitsPerPage, hitsPerPage,
env: context.cloudflare.env env: cloudflare.env
}) })
return { return {
@@ -53,7 +58,7 @@ export default function Route({ loaderData: { data } }: Route.ComponentProps) {
</div> </div>
<Await resolve={data}> <Await resolve={data}>
{({ hits, page, hitsPerPage, totalHits }) => { {({ hits, page = 1, hitsPerPage, totalHits }) => {
return ( return (
<DataTable <DataTable
sort={[{ id: 'create_date', desc: true }]} sort={[{ id: 'create_date', desc: true }]}

View File

@@ -5,7 +5,6 @@ import { useRequest, useToggle } from 'ahooks'
import { import {
AlertTriangleIcon, AlertTriangleIcon,
BanIcon, BanIcon,
CalendarClockIcon,
CalendarIcon, CalendarIcon,
CircleXIcon, CircleXIcon,
ClockAlertIcon, ClockAlertIcon,

View File

@@ -19,7 +19,6 @@ import {
import { request as req } from '@repo/util/request' import { request as req } from '@repo/util/request'
import { Button } from '@repo/ui/components/ui/button' import { Button } from '@repo/ui/components/ui/button'
import type { User } from '../_.$orgid.users.$id/route'
export async function loader({ params, request, context }: Route.LoaderArgs) { export async function loader({ params, request, context }: Route.LoaderArgs) {
const { id } = params const { id } = params
@@ -33,7 +32,7 @@ export async function loader({ params, request, context }: Route.LoaderArgs) {
} }
export default function Route({ loaderData: { data } }: Route.ComponentProps) { export default function Route({ loaderData: { data } }: Route.ComponentProps) {
const { user } = useOutletContext() as { user: User } const { user } = useOutletContext()
return ( return (
<Suspense fallback={<Skeleton />}> <Suspense fallback={<Skeleton />}>

View File

@@ -34,9 +34,8 @@ export function meta() {
} }
export async function loader({ params, request, context }: Route.LoaderArgs) { export async function loader({ params, request, context }: Route.LoaderArgs) {
const { id } = params
const r = await req({ const r = await req({
url: `/users/${id}`, url: `/users/${params.id}`,
request, request,
context context
}) })

View File

@@ -5,6 +5,7 @@ import { Suspense, useState } from 'react'
import { Await, Link, useParams, useSearchParams } from 'react-router' import { Await, Link, useParams, useSearchParams } from 'react-router'
import { MeiliSearchFilterBuilder } from 'meilisearch-helper' import { MeiliSearchFilterBuilder } from 'meilisearch-helper'
import { cloudflareContext } from '@repo/auth/context'
import { DataTable } from '@repo/ui/components/data-table' import { DataTable } from '@repo/ui/components/data-table'
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'
@@ -27,14 +28,18 @@ export function meta({}: Route.MetaArgs) {
} }
export async function loader({ params, context, request }: Route.LoaderArgs) { export async function loader({ params, context, request }: Route.LoaderArgs) {
const cloudflare = context.get(cloudflareContext)
const { searchParams } = new URL(request.url) const { searchParams } = new URL(request.url)
const { orgid } = params
const query = searchParams.get('q') || '' const query = searchParams.get('q') || ''
const sort = searchParams.get('sort') || 'createDate:desc' const sort = searchParams.get('sort') || 'createDate:desc'
const page = Number(searchParams.get('p')) + 1 const page = Number(searchParams.get('p')) + 1
const hitsPerPage = Number(searchParams.get('perPage')) || 25 const hitsPerPage = Number(searchParams.get('perPage')) || 25
let builder = new MeiliSearchFilterBuilder().where('tenant_id', '=', orgid) let builder = new MeiliSearchFilterBuilder().where(
'tenant_id',
'=',
params.orgid
)
const users = createSearch({ const users = createSearch({
index: 'betaeducacao-prod-users_d2o3r5gmm4it7j', index: 'betaeducacao-prod-users_d2o3r5gmm4it7j',
@@ -43,7 +48,7 @@ export async function loader({ params, context, request }: Route.LoaderArgs) {
query, query,
page, page,
hitsPerPage, hitsPerPage,
env: context.cloudflare.env env: cloudflare.env
}) })
return { data: users } return { data: users }
@@ -66,7 +71,7 @@ export default function Route({ loaderData: { data } }: Route.ComponentProps) {
</div> </div>
<Await resolve={data}> <Await resolve={data}>
{({ hits, page, hitsPerPage, totalHits }) => { {({ hits, page = 1, hitsPerPage, totalHits }) => {
return ( return (
<DataTable <DataTable
sort={[{ id: 'createDate', desc: true }]} sort={[{ id: 'createDate', desc: true }]}

View File

@@ -4,8 +4,10 @@ import { MeiliSearchFilterBuilder } from 'meilisearch-helper'
import { data } from 'react-router' import { data } from 'react-router'
import { createSearch } from '@repo/util/meili' import { createSearch } from '@repo/util/meili'
import { cloudflareContext } from '@repo/auth/context'
export async function loader({ params, context, request }: Route.LoaderArgs) { export async function loader({ params, context, request }: Route.LoaderArgs) {
const cloudflare = context.get(cloudflareContext)
const { searchParams } = new URL(request.url) const { searchParams } = new URL(request.url)
const { orgid } = params const { orgid } = params
const query = searchParams.get('q') || '' const query = searchParams.get('q') || ''
@@ -23,7 +25,7 @@ export async function loader({ params, context, request }: Route.LoaderArgs) {
query, query,
page, page,
hitsPerPage, hitsPerPage,
env: context.cloudflare.env env: cloudflare.env
}) })
return data(r) return data(r)

View File

@@ -20,12 +20,12 @@ import { Toaster } from '@repo/ui/components/ui/sonner'
import { request as req } from '@repo/util/request' import { request as req } from '@repo/util/request'
import { AppSidebar } from '@/components/app-sidebar' import { AppSidebar } from '@/components/app-sidebar'
import { Notification } from '@/components/notification' // import { Notification } from '@/components/notification'
export const middleware: Route.MiddlewareFunction[] = [authMiddleware] export const middleware: Route.MiddlewareFunction[] = [authMiddleware]
export async function loader({ params, context, request }: Route.ActionArgs) { export async function loader({ params, context, request }: Route.ActionArgs) {
const user = context.get(userContext) const user = context.get(userContext)!
const rawCookie = request.headers.get('cookie') || '' const rawCookie = request.headers.get('cookie') || ''
const parsedCookies = cookie.parse(rawCookie) const parsedCookies = cookie.parse(rawCookie)
const { sidebar_state = 'true' } = parsedCookies const { sidebar_state = 'true' } = parsedCookies

View File

@@ -9,7 +9,7 @@ import { request as req } from '@repo/util/request'
export const middleware: Route.MiddlewareFunction[] = [authMiddleware] export const middleware: Route.MiddlewareFunction[] = [authMiddleware]
export async function loader({ context, request }: Route.ActionArgs) { export async function loader({ context, request }: Route.ActionArgs) {
const user = context.get(userContext) const user = context.get(userContext)!
const r = await req({ const r = await req({
url: `/users/${user.sub}/orgs`, url: `/users/${user.sub}/orgs`,

View File

@@ -3,24 +3,29 @@ import type { Route } from './+types/route'
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 } from '@repo/auth/context' 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 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(`[${requestId}] Redirecting the user to ${returnTo}`)

View File

@@ -5,12 +5,14 @@ import type { OAuth2Strategy } from 'remix-auth-oauth2'
import { createAuth, type User } from '@repo/auth/auth' import { createAuth, type User } from '@repo/auth/auth'
import { createSessionStorage } from '@repo/auth/session' 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) {

View File

@@ -1,7 +1,6 @@
import type { Route } from './+types/route' import type { Route } from './+types/route'
import type { User } from '@repo/auth/auth' import { userContext, cloudflareContext } from '@repo/auth/context'
import { userContext } from '@repo/auth/context'
import { authMiddleware } from '@repo/auth/middleware/auth' import { authMiddleware } from '@repo/auth/middleware/auth'
export const middleware: Route.MiddlewareFunction[] = [authMiddleware] export const middleware: Route.MiddlewareFunction[] = [authMiddleware]
@@ -13,8 +12,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,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>