| import { env } from "$env/dynamic/private"; |
| import { env as envPublic } from "$env/dynamic/public"; |
| import type { Handle, HandleServerError } from "@sveltejs/kit"; |
| import { collections } from "$lib/server/database"; |
| import { base } from "$app/paths"; |
| import { findUser, refreshSessionCookie, requiresUser } from "$lib/server/auth"; |
| import { ERROR_MESSAGES } from "$lib/stores/errors"; |
| import { sha256 } from "$lib/utils/sha256"; |
| import { addWeeks } from "date-fns"; |
| import { checkAndRunMigrations } from "$lib/migrations/migrations"; |
| import { building } from "$app/environment"; |
| import { refreshAssistantsCounts } from "$lib/assistantStats/refresh-assistants-counts"; |
| import { logger } from "$lib/server/logger"; |
| import { AbortedGenerations } from "$lib/server/abortedGenerations"; |
| import { MetricsServer } from "$lib/server/metrics"; |
|
|
| |
| if (!building) { |
| await checkAndRunMigrations(); |
| if (env.ENABLE_ASSISTANTS) { |
| refreshAssistantsCounts(); |
| } |
|
|
| |
| MetricsServer.getInstance(); |
|
|
| |
| AbortedGenerations.getInstance(); |
| } |
|
|
| export const handleError: HandleServerError = async ({ error, event }) => { |
| |
|
|
| if (building) { |
| throw error; |
| } |
|
|
| if (event.route.id === null) { |
| return { |
| message: `Page ${event.url.pathname} not found`, |
| }; |
| } |
|
|
| const errorId = crypto.randomUUID(); |
|
|
| logger.error({ |
| locals: event.locals, |
| url: event.request.url, |
| params: event.params, |
| request: event.request, |
| error, |
| errorId, |
| }); |
|
|
| return { |
| message: "An error occurred", |
| errorId, |
| }; |
| }; |
|
|
| export const handle: Handle = async ({ event, resolve }) => { |
| logger.debug({ |
| locals: event.locals, |
| url: event.url.pathname, |
| params: event.params, |
| request: event.request, |
| }); |
|
|
| if (event.url.pathname.startsWith(`${base}/api/`) && env.EXPOSE_API !== "true") { |
| return new Response("API is disabled", { status: 403 }); |
| } |
|
|
| function errorResponse(status: number, message: string) { |
| const sendJson = |
| event.request.headers.get("accept")?.includes("application/json") || |
| event.request.headers.get("content-type")?.includes("application/json"); |
| return new Response(sendJson ? JSON.stringify({ error: message }) : message, { |
| status, |
| headers: { |
| "content-type": sendJson ? "application/json" : "text/plain", |
| }, |
| }); |
| } |
|
|
| if (event.url.pathname.startsWith(`${base}/admin/`) || event.url.pathname === `${base}/admin`) { |
| const ADMIN_SECRET = env.ADMIN_API_SECRET || env.PARQUET_EXPORT_SECRET; |
|
|
| if (!ADMIN_SECRET) { |
| return errorResponse(500, "Admin API is not configured"); |
| } |
|
|
| if (event.request.headers.get("Authorization") !== `Bearer ${ADMIN_SECRET}`) { |
| return errorResponse(401, "Unauthorized"); |
| } |
| } |
|
|
| const token = event.cookies.get(env.COOKIE_NAME); |
|
|
| let secretSessionId: string; |
| let sessionId: string; |
|
|
| if (token) { |
| secretSessionId = token; |
| sessionId = await sha256(token); |
|
|
| const user = await findUser(sessionId); |
|
|
| if (user) { |
| event.locals.user = user; |
| } |
| } else { |
| |
| secretSessionId = crypto.randomUUID(); |
| sessionId = await sha256(secretSessionId); |
|
|
| if (await collections.sessions.findOne({ sessionId })) { |
| return errorResponse(500, "Session ID collision"); |
| } |
| } |
|
|
| event.locals.sessionId = sessionId; |
|
|
| |
| const requestContentType = event.request.headers.get("content-type")?.split(";")[0] ?? ""; |
| |
| const nativeFormContentTypes = [ |
| "multipart/form-data", |
| "application/x-www-form-urlencoded", |
| "text/plain", |
| ]; |
|
|
| if (event.request.method === "POST") { |
| refreshSessionCookie(event.cookies, event.locals.sessionId); |
|
|
| if (nativeFormContentTypes.includes(requestContentType)) { |
| const origin = event.request.headers.get("origin"); |
|
|
| if (!origin) { |
| return errorResponse(403, "Non-JSON form requests need to have an origin"); |
| } |
|
|
| const validOrigins = [ |
| new URL(event.request.url).host, |
| ...(envPublic.PUBLIC_ORIGIN ? [new URL(envPublic.PUBLIC_ORIGIN).host] : []), |
| ]; |
|
|
| if (!validOrigins.includes(new URL(origin).host)) { |
| return errorResponse(403, "Invalid referer for POST request"); |
| } |
| } |
| } |
|
|
| if (event.request.method === "POST") { |
| |
| refreshSessionCookie(event.cookies, secretSessionId); |
|
|
| await collections.sessions.updateOne( |
| { sessionId }, |
| { $set: { updatedAt: new Date(), expiresAt: addWeeks(new Date(), 2) } } |
| ); |
| } |
|
|
| if ( |
| !event.url.pathname.startsWith(`${base}/login`) && |
| !event.url.pathname.startsWith(`${base}/admin`) && |
| !["GET", "OPTIONS", "HEAD"].includes(event.request.method) |
| ) { |
| if ( |
| !event.locals.user && |
| requiresUser && |
| !((env.MESSAGES_BEFORE_LOGIN ? parseInt(env.MESSAGES_BEFORE_LOGIN) : 0) > 0) |
| ) { |
| return errorResponse(401, ERROR_MESSAGES.authOnly); |
| } |
|
|
| |
| |
| |
| if ( |
| !requiresUser && |
| !event.url.pathname.startsWith(`${base}/settings`) && |
| !!envPublic.PUBLIC_APP_DISCLAIMER |
| ) { |
| const hasAcceptedEthicsModal = await collections.settings.countDocuments({ |
| sessionId: event.locals.sessionId, |
| ethicsModalAcceptedAt: { $exists: true }, |
| }); |
|
|
| if (!hasAcceptedEthicsModal) { |
| return errorResponse(405, "You need to accept the welcome modal first"); |
| } |
| } |
| } |
|
|
| let replaced = false; |
|
|
| const response = await resolve(event, { |
| transformPageChunk: (chunk) => { |
| |
| if (replaced || !chunk.html.includes("%gaId%")) { |
| return chunk.html; |
| } |
| replaced = true; |
|
|
| return chunk.html.replace("%gaId%", envPublic.PUBLIC_GOOGLE_ANALYTICS_ID); |
| }, |
| }); |
|
|
| return response; |
| }; |
|
|