fix caching

This commit is contained in:
2025-04-04 02:12:45 +02:00
parent 21b04afd10
commit 1e16b4c01d
7 changed files with 166 additions and 144 deletions

View File

@@ -1,7 +1,7 @@
'use client'
import type React from 'react'
import { useRef } from 'react'
import { type ComponentPropsWithoutRef, Fragment, useRef } from 'react'
import { Badge } from '@/components/ui/badge'
import { Button } from '@/components/ui/button'
@@ -68,15 +68,13 @@ export function LeaderboardPage() {
// Get the current leaderboard based on selected tab
const currentLeaderboard =
leaderboardType === 'ranked'
? rankedLeaderboard.alltime
: vanillaLeaderboard.alltime
leaderboardType === 'ranked' ? rankedLeaderboard : vanillaLeaderboard
// Filter leaderboard by search query
const filteredLeaderboard = currentLeaderboard.filter((entry) =>
entry.name.toLowerCase().includes(searchQuery.toLowerCase())
)
console.log(filteredLeaderboard)
// Sort leaderboard
const sortedLeaderboard = [...filteredLeaderboard].sort((a, b) => {
// biome-ignore lint/style/useSingleVarDeclarator: <explanation>
@@ -85,8 +83,8 @@ export function LeaderboardPage() {
// Handle special case for rank which is already sorted
if (sortColumn === 'rank') {
valueA = a.data.rank
valueB = b.data.rank
valueA = a.rank
valueB = b.rank
} else if (sortColumn === 'name') {
valueA = a.name.toLowerCase()
valueB = b.name.toLowerCase()
@@ -94,8 +92,8 @@ export function LeaderboardPage() {
? valueA.localeCompare(valueB)
: valueB.localeCompare(valueA)
} else {
valueA = a.data[sortColumn as keyof typeof a.data] as number
valueB = b.data[sortColumn as keyof typeof b.data] as number
valueA = a[sortColumn as keyof typeof a] as number
valueB = b[sortColumn as keyof typeof b] as number
}
return sortDirection === 'asc' ? valueA - valueB : valueB - valueA
@@ -118,12 +116,11 @@ export function LeaderboardPage() {
if (rank === 3) return <Medal className='h-5 w-5 text-amber-700' />
return null
}
console.log(currentLeaderboard)
return (
<div className='min-h-screen bg-gradient-to-b from-slate-50 to-slate-100 dark:from-slate-950 dark:to-slate-900'>
<div className='container mx-auto px-4 py-8'>
<Card className='overflow-hidden border-none bg-white py-0 shadow-lg dark:bg-slate-900'>
<div className='flex min-h-screen flex-col bg-gradient-to-b from-slate-50 to-slate-100 dark:from-slate-950 dark:to-slate-900'>
<div className='container mx-auto flex flex-1 flex-col px-4 py-4'>
<Card className='flex flex-1 flex-col overflow-hidden border-none bg-white p-0 shadow-lg dark:bg-slate-900'>
<CardHeader className='bg-gradient-to-r from-violet-500 to-purple-600 p-6'>
<div className='flex flex-col items-center justify-between gap-4 md:flex-row'>
<div>
@@ -157,12 +154,12 @@ export function LeaderboardPage() {
</div>
</CardHeader>
<CardContent className='p-0'>
<CardContent className='flex flex-1 flex-col p-0'>
<Tabs
defaultValue={leaderboardType}
value={leaderboardType}
onValueChange={handleTabChange}
className='p-6'
className='flex flex-1 flex-col p-4 md:p-6'
>
<div className='mb-6 flex flex-col items-start justify-between gap-4 sm:flex-row sm:items-center'>
<TabsList className='bg-slate-100 dark:bg-slate-800'>
@@ -181,7 +178,7 @@ export function LeaderboardPage() {
</div>
</div>
<TabsContent value='ranked' className='m-0'>
<TabsContent value='ranked' className='m-0 flex flex-1 flex-col'>
<LeaderboardTable
leaderboard={sortedLeaderboard}
sortColumn={sortColumn}
@@ -192,7 +189,7 @@ export function LeaderboardPage() {
/>
</TabsContent>
<TabsContent value='vanilla' className='m-0'>
<TabsContent value='vanilla' className='m-0 flex flex-1 flex-col'>
<LeaderboardTable
leaderboard={sortedLeaderboard}
sortColumn={sortColumn}
@@ -230,18 +227,20 @@ function LeaderboardTable({
const tableContainerRef = useRef<HTMLDivElement>(null)
// Set a fixed row height for virtualization
const ROW_HEIGHT = 56 // Adjust based on your actual row height
const ROW_HEIGHT = 39 // Adjust based on your actual row height
console.log(leaderboard.length)
// Create virtualizer instance
const rowVirtualizer = useVirtualizer({
count: leaderboard.length,
getScrollElement: () => tableContainerRef.current,
estimateSize: () => ROW_HEIGHT,
overscan: 10, // Number of items to render before/after the visible area
overscan: 12, // Number of items to render before/after the visible area
})
// Get the virtualized rows
const virtualRows = rowVirtualizer.getVirtualItems()
console.log({ virtualRows })
console.log(rowVirtualizer.getTotalSize())
const paddingTop = virtualRows.length > 0 ? virtualRows?.[0]?.start : 0
const paddingBottom =
virtualRows.length > 0
@@ -249,15 +248,15 @@ function LeaderboardTable({
(virtualRows?.[virtualRows.length - 1]?.end ?? 0)
: 0
return (
<div className='overflow-hidden rounded-lg border'>
<div className='flex flex-1 flex-col overflow-hidden rounded-lg border'>
<div
ref={tableContainerRef}
className='overflow-auto'
style={{ height: '500px', maxHeight: '70vh' }}
className='flex-1 overflow-auto overflow-x-auto'
style={{ maxHeight: 'calc(100vh - 300px)' }}
>
<Table>
<TableHeader className='sticky top-0 z-10 bg-white dark:bg-slate-900'>
<TableRow className='bg-slate-50 dark:bg-slate-800/50'>
<TableRow className=' bg-slate-50 dark:bg-slate-800/50'>
<TableHead className='w-[80px]'>
<SortableHeader
column='rank'
@@ -278,6 +277,7 @@ function LeaderboardTable({
</TableHead>
<TableHead className='text-right'>
<SortableHeader
className='w-full justify-end'
column='mmr'
label='MMR'
currentSort={sortColumn}
@@ -285,8 +285,9 @@ function LeaderboardTable({
onSort={onSort}
/>
</TableHead>
<TableHead className='text-right'>
<TableHead className='text-right' align={'right'}>
<SortableHeader
className='w-full justify-end'
column='peak_mmr'
label='Peak MMR'
currentSort={sortColumn}
@@ -296,6 +297,7 @@ function LeaderboardTable({
</TableHead>
<TableHead className='text-right'>
<SortableHeader
className='w-full justify-end'
column='winrate'
label='Win Rate'
currentSort={sortColumn}
@@ -305,6 +307,7 @@ function LeaderboardTable({
</TableHead>
<TableHead className='text-right'>
<SortableHeader
className='w-full justify-end'
column='wins'
label='Wins'
currentSort={sortColumn}
@@ -314,6 +317,7 @@ function LeaderboardTable({
</TableHead>
<TableHead className='text-right'>
<SortableHeader
className='w-full justify-end'
column='losses'
label='Losses'
currentSort={sortColumn}
@@ -323,6 +327,7 @@ function LeaderboardTable({
</TableHead>
<TableHead className='text-right'>
<SortableHeader
className='w-full justify-end'
column='totalgames'
label='Games'
currentSort={sortColumn}
@@ -332,6 +337,7 @@ function LeaderboardTable({
</TableHead>
<TableHead className='text-right'>
<SortableHeader
className='w-full justify-end'
column='streak'
label='Streak'
currentSort={sortColumn}
@@ -342,96 +348,105 @@ function LeaderboardTable({
</TableRow>
</TableHeader>
<TableBody>
{paddingTop > 0 && (
<tr>
<td style={{ height: `${paddingTop}px` }} colSpan={9} />
</tr>
)}
{leaderboard.length > 0 ? (
leaderboard.map((entry) => {
const winrate = entry.data.winrate * 100
virtualRows.map((virtualRow) => {
const entry = leaderboard[virtualRow.index]
const winrate = entry.winrate * 100
return (
<TableRow
key={entry.id}
className={cn(
'transition-colors hover:bg-slate-50 dark:hover:bg-slate-800/70',
entry.data.rank <= 3 &&
'bg-amber-50/50 dark:bg-amber-950/20'
)}
>
<TableCell className='font-medium'>
<div className='flex items-center gap-1.5'>
{getMedal(entry.data.rank)}
<span>{entry.data.rank}</span>
</div>
</TableCell>
<TableCell>
<Link
href={`/players/${entry.id}`}
className='flex items-center gap-2 hover:underline'
>
{/*<Avatar className='h-8 w-8'>*/}
{/* <AvatarImage*/}
{/* src={`https://cdn.discordapp.com/avatars/${entry.id}/avatar.png`}*/}
{/* alt={entry.name}*/}
{/* />*/}
{/* <AvatarFallback className='bg-violet-100 text-violet-700 text-xs dark:bg-violet-900 dark:text-violet-300'>*/}
{/* {entry.name.slice(0, 2).toUpperCase()}*/}
{/* </AvatarFallback>*/}
{/*</Avatar>*/}
<span className='font-medium'>{entry.name}</span>
{entry.data.streak >= 3 && (
<Badge className='bg-orange-500 text-white'>
<Flame className='mr-1 h-3 w-3' />
Hot Streak
</Badge>
)}
</Link>
</TableCell>
<TableCell className='text-right font-medium'>
{Math.round(entry.data.mmr)}
</TableCell>
<TableCell className='text-right'>
<div className='flex items-center justify-end gap-1'>
<TrendingUp className='h-3.5 w-3.5 text-violet-500' />
{Math.round(entry.data.peak_mmr)}
</div>
</TableCell>
<TableCell className='text-right'>
<Badge
variant='outline'
className={cn(
'font-normal',
winrate > 60
? 'border-emerald-200 bg-emerald-50 text-emerald-700 dark:border-emerald-800 dark:bg-emerald-950 dark:text-emerald-300'
: winrate < 40
? 'border-rose-200 bg-rose-50 text-rose-700 dark:border-rose-800 dark:bg-rose-950 dark:text-rose-300'
: 'border-slate-200 bg-slate-50 text-slate-700 dark:border-slate-800 dark:bg-slate-900 dark:text-slate-300'
)}
>
{Math.round(winrate)}%
</Badge>
</TableCell>
<TableCell className='text-right text-emerald-600 dark:text-emerald-400'>
{entry.data.wins}
</TableCell>
<TableCell className='text-right text-rose-600 dark:text-rose-400'>
{entry.data.losses}
</TableCell>
<TableCell className='text-right text-slate-600 dark:text-slate-400'>
{entry.data.totalgames}
</TableCell>
<TableCell className='text-right'>
{entry.data.streak > 0 ? (
<span className='flex items-center justify-end text-emerald-600 dark:text-emerald-400'>
<ArrowUp className='mr-1 h-3.5 w-3.5' />
{entry.data.streak}
</span>
) : entry.data.streak < 0 ? (
<span className='flex items-center justify-end text-rose-600 dark:text-rose-400'>
<ArrowDown className='mr-1 h-3.5 w-3.5' />
{Math.abs(entry.data.streak)}
</span>
) : (
<span>0</span>
<Fragment key={entry.id}>
{/* Add padding to the top to push content into view */}
<TableRow
className={cn(
'transition-colors hover:bg-slate-50 dark:hover:bg-slate-800/70'
)}
</TableCell>
</TableRow>
>
<TableCell className='font-medium'>
<div className='flex items-center gap-1.5'>
{getMedal(entry.rank)}
<span>{entry.rank}</span>
</div>
</TableCell>
<TableCell>
<Link
href={`/players/${entry.id}`}
className='flex items-center gap-2 hover:underline'
>
{/*<Avatar className='h-8 w-8'>*/}
{/* <AvatarImage*/}
{/* src={`https://cdn.discordapp.com/avatars/${entry.id}/avatar.png`}*/}
{/* alt={entry.name}*/}
{/* />*/}
{/* <AvatarFallback className='bg-violet-100 text-violet-700 text-xs dark:bg-violet-900 dark:text-violet-300'>*/}
{/* {entry.name.slice(0, 2).toUpperCase()}*/}
{/* </AvatarFallback>*/}
{/*</Avatar>*/}
<span className='font-medium'>{entry.name}</span>
{entry.streak >= 3 && (
<Badge className='bg-orange-500 text-white'>
<Flame className='mr-1 h-3 w-3' />
Hot Streak
</Badge>
)}
</Link>
</TableCell>
<TableCell className='text-right font-medium font-mono'>
{Math.round(entry.mmr)}
</TableCell>
<TableCell className='text-right font-mono'>
<div className='flex items-center justify-end gap-1'>
{Math.round(entry.peak_mmr)}
<TrendingUp className='h-3.5 w-3.5 text-violet-500' />
</div>
</TableCell>
<TableCell className='text-right'>
<Badge
variant='outline'
className={cn(
'font-normal ',
winrate > 60
? 'border-emerald-200 bg-emerald-50 text-emerald-700 dark:border-emerald-800 dark:bg-emerald-950 dark:text-emerald-300'
: winrate < 40
? 'border-rose-200 bg-rose-50 text-rose-700 dark:border-rose-800 dark:bg-rose-950 dark:text-rose-300'
: 'border-slate-200 bg-slate-50 text-slate-700 dark:border-slate-800 dark:bg-slate-900 dark:text-slate-300'
)}
>
{Math.round(winrate)}%
</Badge>
</TableCell>
<TableCell className='text-right text-emerald-600 dark:text-emerald-400'>
{entry.wins}
</TableCell>
<TableCell className='text-right text-rose-600 dark:text-rose-400'>
{entry.losses}
</TableCell>
<TableCell className='text-right font-mono text-slate-600 dark:text-slate-400'>
{entry.totalgames}
</TableCell>
<TableCell className='text-right'>
{entry.streak > 0 ? (
<span className='flex items-center justify-end text-emerald-600 dark:text-emerald-400'>
<ArrowUp className='mr-1 h-3.5 w-3.5' />
{entry.streak}
</span>
) : entry.streak < 0 ? (
<span className='flex items-center justify-end font-mono text-rose-600 dark:text-rose-400'>
<ArrowDown className='mr-1 h-3.5 w-3.5' />
<span className={'w-[2ch]'}>
{Math.abs(entry.streak)}
</span>
</span>
) : (
<span>0</span>
)}
</TableCell>
</TableRow>
</Fragment>
)
})
) : (
@@ -441,6 +456,11 @@ function LeaderboardTable({
</TableCell>
</TableRow>
)}
{paddingBottom > 0 && (
<tr>
<td style={{ height: `${paddingBottom}px` }} colSpan={9} />
</tr>
)}
</TableBody>
</Table>
</div>
@@ -448,7 +468,7 @@ function LeaderboardTable({
)
}
interface SortableHeaderProps {
interface SortableHeaderProps extends ComponentPropsWithoutRef<'button'> {
column: string
label: string
currentSort: string
@@ -462,12 +482,19 @@ function SortableHeader({
currentSort,
direction,
onSort,
className,
...rest
}: SortableHeaderProps) {
const isActive = currentSort === column
return (
<button
className='flex items-center gap-1 transition-colors hover:text-violet-600 dark:hover:text-violet-400'
type={'button'}
className={cn(
'flex items-center gap-1 transition-colors hover:text-violet-600 dark:hover:text-violet-400',
className
)}
{...rest}
onClick={() => onSort(column)}
>
{label}

View File

@@ -1,5 +1,4 @@
import { LeaderboardPage } from '@/app/_components/leaderboard'
import { UserStats } from '@/app/_components/user-stats'
import { auth } from '@/server/auth'
import { RANKED_CHANNEL, VANILLA_CHANNEL } from '@/shared/constants'
import { HydrateClient, api } from '@/trpc/server'
@@ -19,11 +18,11 @@ export default async function Home() {
}
return (
<HydrateClient>
<Suspense>
<Suspense>
<HydrateClient>
{/*<UserStats/>*/}
<LeaderboardPage />
</Suspense>
</HydrateClient>
</HydrateClient>
</Suspense>
)
}

View File

@@ -25,7 +25,6 @@ import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'
import { cn } from '@/lib/utils'
import { RANKED_CHANNEL, VANILLA_CHANNEL } from '@/shared/constants'
import { api } from '@/trpc/react'
import { isNotNull } from 'drizzle-orm'
import {
ArrowDownCircle,
ArrowUpCircle,
@@ -84,8 +83,6 @@ export function UserInfo() {
user_id: id,
})
console.log(rankedLeaderboard, vanillaLeaderboard)
// Filter games by leaderboard if needed
const filteredGamesByLeaderboard =
leaderboardFilter === 'all'

View File

@@ -32,7 +32,6 @@ export async function syncHistory() {
.where(eq(metadata.key, 'history_cursor'))
.limit(1)
.then((res) => res[0])
console.log('cursor', cursor)
const data = await ky
.get('https://api.neatqueue.com/api/history/1226193436521267223', {
searchParams: {
@@ -64,9 +63,6 @@ export async function syncHistory() {
value: firstGame,
},
})
console.log('matches', matches)
console.log('firstGame', firstGame)
console.log('data', data)
const chunkedData = chunk(data.data, 100)
for (const chunk of chunkedData) {

View File

@@ -1,9 +1,6 @@
import { createTRPCRouter, publicProcedure } from '@/server/api/trpc'
import { LeaderboardService } from '@/server/services/leaderboard'
import {
type LeaderboardResponse,
neatqueue_service,
} from '@/server/services/neatqueue.service'
import type { LeaderboardResponse } from '@/server/services/neatqueue.service'
import { z } from 'zod'
const service = new LeaderboardService()
@@ -17,7 +14,7 @@ export const leaderboard_router = createTRPCRouter({
.query(async ({ input }) => {
return (await service.getLeaderboard(
input.channel_id
)) as LeaderboardResponse
)) as LeaderboardResponse['alltime']
}),
get_user_rank: publicProcedure
.input(

View File

@@ -21,18 +21,12 @@ export class LeaderboardService {
const pipeline = redis.pipeline()
pipeline.del(zsetKey) // clear existing
for (const entry of fresh.alltime) {
for (const entry of fresh) {
// store by mmr for ranking
pipeline.zadd(zsetKey, entry.data.mmr, entry.id)
pipeline.zadd(zsetKey, entry.rank, entry.id)
// store user data separately for quick lookups
pipeline.hset(`user:${entry.id}`, {
name: entry.name,
mmr: entry.data.mmr,
wins: entry.data.wins,
losses: entry.data.losses,
// add other fields you need for quick lookup
})
pipeline.hset(`user:${entry.id}`, entry)
}
pipeline.expire(zsetKey, 180)

View File

@@ -11,11 +11,23 @@ const BMM_SERVER_ID = '1226193436521267223'
export const neatqueue_service = {
get_leaderboard: async (channel_id: string) => {
const response = await instance.get(
`leaderboard/${BMM_SERVER_ID}/${channel_id}`
)
const res = await instance
.get(`leaderboard/${BMM_SERVER_ID}/${channel_id}`)
.json<LeaderboardResponse>()
return response.json<LeaderboardResponse>()
//desc
res.alltime.sort((a, b) => b.data.mmr - a.data.mmr)
const fixed: Array<Data & { id: string; name: string }> = res.alltime.map(
(entry, idx) => {
return {
...entry.data,
rank: idx + 1,
id: entry.id,
name: entry.name,
}
}
)
return fixed
},
get_history: async (
player_ids: string[],