Toothless' TSS Bot
Coming soon
TSS analytics are getting tucked away for a little bit. Check back soon.
import path from 'node:path' import { defineConfig, loadEnv } from 'vite' import react from '@vitejs/plugin-react' import tailwindcss from '@tailwindcss/vite' import obfuscatorPlugin from 'rollup-plugin-obfuscator' const OBFUSCATOR_OPTIONS = { compact: true, controlFlowFlattening: false, deadCodeInjection: false, debugProtection: false, disableConsoleOutput: false, identifierNamesGenerator: 'hexadecimal', log: false, numbersToExpressions: false, renameGlobals: false, selfDefending: false, simplify: true, splitStrings: false, stringArray: true, stringArrayCallsTransform: false, stringArrayEncoding: ['base64'], stringArrayRotate: true, stringArrayShuffle: true, stringArrayThreshold: 0.75, transformObjectKeys: false, unicodeEscapeSequence: false, } function obfuscate() { const factory = obfuscatorPlugin.default || obfuscatorPlugin const inner = factory({ global: true, options: OBFUSCATOR_OPTIONS }) return { ...inner, apply: 'build', enforce: 'post' } } const MAX_TEAM_NAME_LENGTH = 80 function isAllowedApiUrl(req) { const url = new URL(req.url, 'http://localhost') const params = url.searchParams if (url.pathname === '/api/viewers' && (req.method === 'GET' || req.method === 'HEAD')) return true if (url.pathname === '/api/viewers/event' && req.method === 'POST') return true if (req.method !== 'GET' && req.method !== 'HEAD') return false if (url.pathname === '/api/tss/leaderboard/teams') { const keys = [...params.keys()] const limit = Number(params.get('limit') || 100) return keys.every((key) => key === 'limit') && Number.isInteger(limit) && limit >= 1 && limit <= 100 } if (url.pathname === '/api/tss/games/recent') { const keys = [...params.keys()] const limit = Number(params.get('limit') || 50) return keys.every((key) => key === 'limit') && Number.isInteger(limit) && limit >= 1 && limit <= 100 } if (url.pathname === '/api/tss/teams/resolve') { const keys = [...params.keys()] const name = params.get('name') || '' return keys.every((key) => key === 'name') && name.length >= 2 && name.length <= MAX_TEAM_NAME_LENGTH } if (url.pathname === '/api/tss/teams/search') { const keys = [...params.keys()] const query = params.get('q') || params.get('name') || '' const limit = Number(params.get('limit') || 10) return ( keys.every((key) => ['q', 'name', 'limit'].includes(key)) && query.length >= 2 && query.length <= MAX_TEAM_NAME_LENGTH && Number.isInteger(limit) && limit >= 1 && limit <= 20 ) } if ([...params.keys()].length) return false try { const match = url.pathname.match(/^\/api\/tss\/teams\/([^/]+)(?:\/(history|games))?$/) const teamName = match ? decodeURIComponent(match[1]) : '' return Boolean(teamName) && teamName.length <= MAX_TEAM_NAME_LENGTH } catch { return false } } function apiGuard() { return { name: 'api-guard', configureServer(server) { server.middlewares.use(async (req, res, next) => { if (!req.url?.startsWith('/api/')) { next() return } if (isAllowedApiUrl(req)) { next() return } res.writeHead(404, { 'content-type': 'application/json; charset=utf-8', 'x-content-type-options': 'nosniff', }) res.end(JSON.stringify({ error: 'API route not found' })) }) }, } } function comingSoonHtml() { return `
Toothless' TSS Bot
TSS analytics are getting tucked away for a little bit. Check back soon.