| import { config, ready } from "$lib/server/config"; |
| import type { Handle, HandleServerError, ServerInit, HandleFetch } from "@sveltejs/kit"; |
| import { collections } from "$lib/server/database"; |
| import { base } from "$app/paths"; |
| import { authenticateRequest, refreshSessionCookie, requiresUser } from "$lib/server/auth"; |
| import { ERROR_MESSAGES } from "$lib/stores/errors"; |
| import { addWeeks } from "date-fns"; |
| import { checkAndRunMigrations } from "$lib/migrations/migrations"; |
| import { building, dev } from "$app/environment"; |
| import { logger } from "$lib/server/logger"; |
| import { AbortedGenerations } from "$lib/server/abortedGenerations"; |
| import { MetricsServer } from "$lib/server/metrics"; |
| import { initExitHandler } from "$lib/server/exitHandler"; |
| import { refreshAssistantsCounts } from "$lib/jobs/refresh-assistants-counts"; |
| import { refreshConversationStats } from "$lib/jobs/refresh-conversation-stats"; |
| import { adminTokenManager } from "$lib/server/adminToken"; |
| import { isHostLocalhost } from "$lib/server/isURLLocal"; |
|
|
| export const init: ServerInit = async () => { |
| |
| await ready; |
|
|
| |
| if (!building) { |
| |
| process.env.HF_TOKEN ??= config.HF_TOKEN; |
|
|
| logger.info("Starting server..."); |
| initExitHandler(); |
|
|
| checkAndRunMigrations(); |
| if (config.ENABLE_ASSISTANTS) { |
| refreshAssistantsCounts(); |
| } |
| refreshConversationStats(); |
|
|
| |
| MetricsServer.getInstance(); |
|
|
| |
| AbortedGenerations.getInstance(); |
|
|
| adminTokenManager.displayToken(); |
|
|
| if (config.EXPOSE_API) { |
| logger.warn( |
| "The EXPOSE_API flag has been deprecated. The API is now required for chat-ui to work." |
| ); |
| } |
| } |
| }; |
|
|
| export const handleError: HandleServerError = async ({ error, event, status, message }) => { |
| |
|
|
| 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, |
| message, |
| error, |
| errorId, |
| status, |
| stack: error instanceof Error ? error.stack : undefined, |
| }); |
|
|
| return { |
| message: "An error occurred", |
| errorId, |
| }; |
| }; |
|
|
| export const handle: Handle = async ({ event, resolve }) => { |
| await ready.then(() => { |
| config.checkForUpdates(); |
| }); |
|
|
| logger.debug({ |
| locals: event.locals, |
| url: event.url.pathname, |
| params: event.params, |
| request: event.request, |
| }); |
|
|
| 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 = config.ADMIN_API_SECRET || config.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 auth = await authenticateRequest( |
| { type: "svelte", value: event.request.headers }, |
| { type: "svelte", value: event.cookies } |
| ); |
|
|
| event.locals.user = auth.user || undefined; |
| event.locals.sessionId = auth.sessionId; |
|
|
| event.locals.isAdmin = |
| event.locals.user?.isAdmin || adminTokenManager.isAdmin(event.locals.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") { |
| 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, |
| ...(config.PUBLIC_ORIGIN ? [new URL(config.PUBLIC_ORIGIN).host] : []), |
| ]; |
|
|
| if (!validOrigins.includes(new URL(origin).host)) { |
| return errorResponse(403, "Invalid referer for POST request"); |
| } |
| } |
| } |
|
|
| if ( |
| event.request.method === "POST" || |
| event.url.pathname.startsWith(`${base}/login`) || |
| event.url.pathname.startsWith(`${base}/login/callback`) |
| ) { |
| |
| refreshSessionCookie(event.cookies, auth.secretSessionId); |
|
|
| await collections.sessions.updateOne( |
| { sessionId: auth.sessionId }, |
| { $set: { updatedAt: new Date(), expiresAt: addWeeks(new Date(), 2) } } |
| ); |
| } |
|
|
| if ( |
| !event.url.pathname.startsWith(`${base}/login`) && |
| !event.url.pathname.startsWith(`${base}/admin`) && |
| !event.url.pathname.startsWith(`${base}/settings`) && |
| !["GET", "OPTIONS", "HEAD"].includes(event.request.method) |
| ) { |
| if ( |
| !event.locals.user && |
| requiresUser && |
| !((config.MESSAGES_BEFORE_LOGIN ? parseInt(config.MESSAGES_BEFORE_LOGIN) : 0) > 0) |
| ) { |
| return errorResponse(401, ERROR_MESSAGES.authOnly); |
| } |
|
|
| |
| |
| |
| if ( |
| !requiresUser && |
| !event.url.pathname.startsWith(`${base}/settings`) && |
| config.PUBLIC_APP_DISCLAIMER === "1" |
| ) { |
| 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%", config.PUBLIC_GOOGLE_ANALYTICS_ID); |
| }, |
| filterSerializedResponseHeaders: (header) => { |
| return header.includes("content-type"); |
| }, |
| }); |
|
|
| |
| if (config.ALLOW_IFRAME !== "true") { |
| response.headers.append("Content-Security-Policy", "frame-ancestors 'none';"); |
| } |
|
|
| if ( |
| event.url.pathname.startsWith(`${base}/login/callback`) || |
| event.url.pathname.startsWith(`${base}/login`) |
| ) { |
| response.headers.append("Cache-Control", "no-store"); |
| } |
|
|
| if (event.url.pathname.startsWith(`${base}/api/`)) { |
| |
| const requestOrigin = event.request.headers.get("origin"); |
|
|
| |
| let allowedOrigin = config.PUBLIC_ORIGIN ? new URL(config.PUBLIC_ORIGIN).origin : undefined; |
|
|
| if ( |
| dev || |
| !requestOrigin || |
| isHostLocalhost(new URL(requestOrigin).hostname) |
| ) { |
| allowedOrigin = "*"; |
| } else if (allowedOrigin === requestOrigin) { |
| allowedOrigin = requestOrigin; |
| } |
|
|
| allowedOrigin = "*"; |
|
|
| if (allowedOrigin) { |
| response.headers.set("Access-Control-Allow-Origin", allowedOrigin); |
| response.headers.set( |
| "Access-Control-Allow-Methods", |
| "GET, POST, PUT, PATCH, DELETE, OPTIONS" |
| ); |
| response.headers.set("Access-Control-Allow-Headers", "Content-Type, Authorization"); |
| } |
| } |
| return response; |
| }; |
|
|
| export const handleFetch: HandleFetch = async ({ event, request, fetch }) => { |
| if (isHostLocalhost(new URL(request.url).hostname)) { |
| const cookieHeader = event.request.headers.get("cookie"); |
| if (cookieHeader) { |
| const headers = new Headers(request.headers); |
| headers.set("cookie", cookieHeader); |
|
|
| return fetch(new Request(request, { headers })); |
| } |
| } |
|
|
| return fetch(request); |
| }; |
|
|