sendou.ink/app/features/api-public/api-public-utils.server.ts
2025-11-09 11:07:20 +02:00

35 lines
1009 B
TypeScript

import { cors } from "remix-utils/cors";
import * as ApiRepository from "~/features/api/ApiRepository.server";
export async function loadApiTokensCache() {
const envTokens = process.env.PUBLIC_API_TOKENS?.split(",") ?? [];
const dbTokens = await ApiRepository.allApiTokens();
return new Set([...envTokens, ...dbTokens]);
}
let apiTokens = await loadApiTokensCache();
export async function refreshApiTokensCache() {
apiTokens = await loadApiTokensCache();
}
export function requireBearerAuth(req: Request) {
const authHeader = req.headers.get("Authorization");
if (!authHeader) {
throw new Response("Missing Authorization header", { status: 401 });
}
const token = authHeader.replace("Bearer ", "");
if (!apiTokens.has(token)) {
throw new Response("Invalid token", { status: 401 });
}
}
export async function handleOptionsRequest(req: Request) {
if (req.method === "OPTIONS") {
throw await cors(req, new Response("OK", { status: 204 }), {
origin: "*",
credentials: true,
});
}
}