Files
Ivan Vasilov 56de26fe22 chore: Migrate the monorepo to use Tailwind v4 (#45318)
This PR migrates the whole monorepo to use Tailwind v4:
- Removed `@tailwindcss/container-queries` plugin since it's included by
default in v4,
- Bump all instances of Tailwind to v4. Made minimal changes to the
shared config to remove non-supported features (`alpha` mentions),
- Migrate all apps to be compatible with v4 configs,
- Fix the `typography.css` import in 3 apps,
- Add missing rules which were included by default in v3,
- Run `pnpm dlx @tailwindcss/upgrade` on all apps, which renames a lot
of classes
- Rename all misnamed classes according to
https://tailwindcss.com/docs/upgrade-guide#renamed-utilities in all
apps.

---------

Co-authored-by: Jordi Enric <jordi.err@gmail.com>
2026-04-30 10:53:24 +00:00

195 lines
7.1 KiB
TypeScript

import { useParams } from 'common'
import { ExternalLink, RefreshCw } from 'lucide-react'
import Link from 'next/link'
import { useQueryState } from 'nuqs'
import { useEffect } from 'react'
import { Button, cn, CriticalIcon, Separator } from 'ui'
import { Admonition, TimestampInfo } from 'ui-patterns'
import { GenericSkeletonLoader } from 'ui-patterns/ShimmeringLoader'
import { UserHeader } from './UserHeader'
import { PANEL_PADDING } from './Users.constants'
import { LOGS_TABLES } from '@/components/interfaces/Settings/Logs/Logs.constants'
import { ButtonTooltip } from '@/components/ui/ButtonTooltip'
import { User } from '@/data/auth/users-infinite-query'
import useLogsPreview from '@/hooks/analytics/useLogsPreview'
import { useLogsUrlState } from '@/hooks/analytics/useLogsUrlState'
interface UserLogsProps {
user: User
}
const API_LOGS_QUERY = (userId: string) =>
`select\n cast(timestamp as datetime) as timestamp,\n event_message, metadata \nfrom edge_logs \nWHERE (\n metadata[SAFE_OFFSET(0)].request[SAFE_OFFSET(0)].sb[SAFE_OFFSET(0)].auth_user\n = '${userId}'\n)\nlimit 100`
export const UserLogs = ({ user }: UserLogsProps) => {
const { ref } = useParams()
const { filters, setFilters } = useLogsUrlState()
const [, setFiltersValue] = useQueryState('f')
const {
logData: authLogs,
isSuccess: isSuccessAuthLogs,
isLoading: isLoadingAuthLogs,
refresh,
} = useLogsPreview({
projectRef: ref as string,
table: LOGS_TABLES.auth,
filterOverride: { search_query: user.id },
limit: 5,
})
useEffect(() => {
if (user.id) setFilters({ ...filters, search_query: user.id })
return () => {
setFiltersValue(null)
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [user.id])
return (
<div>
<UserHeader user={user} />
<Separator />
<div className={cn('flex flex-col gap-y-3', PANEL_PADDING)}>
<div>
<p>API logs</p>
<p className="text-sm text-foreground-light">
View edge logs for requests made by this user
</p>
</div>
<Button asChild type="default" className="w-min">
<Link
href={`/project/${ref}/logs/explorer?q=${encodeURIComponent(API_LOGS_QUERY(user.id ?? ''))}`}
>
Open in Log Explorer
</Link>
</Button>
</div>
<Separator />
<div className={cn('flex flex-col gap-y-3', PANEL_PADDING)}>
<div>
<p>Authentication logs</p>
<p className="text-sm text-foreground-light">
Latest logs from authentication for this user in the past hour
</p>
</div>
{/* [Joshen] This whole thing here i reckon we can shift to a component, if in the future we wanna add more user logs */}
<div className="flex items-center justify-between">
<div className="flex items-center">
<Button
type={'status_code' in filters ? 'default' : 'secondary'}
className="rounded-r-none border-r-0"
disabled={isLoadingAuthLogs}
onClick={() => setFilters({ search_query: user.id })}
>
Show all
</Button>
<div className="border-button border border-l-0 py-3" />
<Button
type={'status_code' in filters ? 'secondary' : 'default'}
className="rounded-l-none border-l-0"
disabled={isLoadingAuthLogs}
onClick={() =>
setFilters({
search_query: user.id,
status_code: { client_error: true, server_error: true },
})
}
>
Error only
</Button>
</div>
<Button
type="default"
loading={isLoadingAuthLogs}
disabled={isLoadingAuthLogs}
icon={<RefreshCw />}
onClick={() => refresh()}
>
Refresh
</Button>
</div>
{isLoadingAuthLogs && !isSuccessAuthLogs ? (
<GenericSkeletonLoader />
) : authLogs.length === 0 ? (
<Admonition
type="note"
title="No authentication logs available for this user"
description="Auth events such as logging in will be shown here"
/>
) : (
<div>
<div className="border border-b-0 rounded-t-md divide-y overflow-hidden">
{authLogs.map((log) => {
const status = ((log.status ?? '-') as any).toString()
const is400 = status.startsWith('4')
const is500 = status.startsWith('5')
return (
<div
key={log.id}
className="flex items-center transition font-mono px-2 py-1.5 bg-surface-100 divide-x"
>
<p className="text-xs text-foreground-light min-w-[125px] w-[125px] px-1">
<TimestampInfo utcTimestamp={log.timestamp / 1000} />
</p>
<div className="flex items-center text-xs text-foreground-light h-[22px] min-w-[70px] w-[70px] px-2">
<div
className={cn(
'flex items-center justify-center gap-x-1',
!!log.status && 'border px-1 py-0.5 rounded-sm',
is400
? 'text-warning border-warning bg-warning-300'
: is500
? 'text-destructive border-destructive bg-destructive-300'
: ''
)}
>
{(is400 || is500) && (
<CriticalIcon hideBackground className={cn(is400 && 'text-warning')} />
)}
{status}
</div>
</div>
<p className="group relative flex items-center py-1.5 text-xs text-foreground-light px-2 truncate w-full">
{`${log.path} | ${log.msg}`}
<ButtonTooltip
type="outline"
asChild
tooltip={{ content: { text: 'Open in logs' } }}
className="px-1.5 absolute right-0 top-0 opacity-0 group-hover:opacity-100 transition bg-background focus-visible:opacity-100"
>
<Link href={`/project/${ref}/logs/auth-logs?log=${log.id}`}>
<ExternalLink size="12" className="text-foreground-light" />
</Link>
</ButtonTooltip>
</p>
</div>
)
})}
</div>
<Button
block
asChild
type="outline"
className="transition rounded-t-none text-foreground-light hover:text-foreground"
>
<Link href={`/project/${ref}/logs/auth-logs?s=${user.id}`}>See more logs</Link>
</Button>
</div>
)}
</div>
</div>
)
}