| | 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; |
| | } |
| |
|
| | 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); |
| | }; |
| |
|