mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-06-19 11:35:47 +00:00
Some checks are pending
Docker Dev Images Build / build (Dockerfile, librechat-dev, node) (push) Waiting to run
Docker Dev Images Build / build (Dockerfile.multi, librechat-dev-api, api-build) (push) Waiting to run
GitNexus Index / index (push) Waiting to run
GitNexus Index / post-index (push) Blocked by required conditions
Sync Locize Translations & Create Translation PR / Sync Translation Keys with Locize (push) Waiting to run
Sync Locize Translations & Create Translation PR / Create Translation PR on Version Published (push) Blocked by required conditions
Sync Helm Chart Tags / Ignore non-main push (push) Waiting to run
Sync Helm Chart Tags / Sync chart tags (push) Waiting to run
* 🛟 fix: Auto-Recover from Stale Service Worker Assets After Deploys - 404 missing static assets in the SPA fallback instead of serving index.html - inline recovery script unregisters stale SWs and reloads once on chunk failure - route vite:preloadError into the same recovery path for stale lazy chunks * 🛟 fix: Address Review — SW-Side Recovery, Scoped Unregister, Shared Fallback - importScripts'd sw-heal.js pings window clients on activation and reloads ones that can't pong: stale pages carry no recovery code of their own - scope SW unregistration to the deployment base for subpath installs - preventDefault vite:preloadError only when a recovery reload was initiated - extract createSpaFallback and apply the asset 404 guard to experimental.js
65 lines
2.2 KiB
JavaScript
65 lines
2.2 KiB
JavaScript
const path = require('path');
|
|
const express = require('express');
|
|
const expressStaticGzip = require('express-static-gzip');
|
|
|
|
const oneDayInSeconds = 24 * 60 * 60;
|
|
|
|
const sMaxAge = process.env.STATIC_CACHE_S_MAX_AGE || oneDayInSeconds;
|
|
const maxAge = process.env.STATIC_CACHE_MAX_AGE || oneDayInSeconds * 2;
|
|
const isEnabled = (value) => value === true || String(value).toLowerCase() === 'true';
|
|
|
|
/**
|
|
* Creates an Express static middleware with optional precompressed asset serving and configurable caching
|
|
*
|
|
* @param {string} staticPath - The file system path to serve static files from
|
|
* @param {Object} [options={}] - Configuration options
|
|
* @param {boolean} [options.noCache=false] - If true, disables caching entirely for all files
|
|
* @param {boolean} [options.skipGzipScan=false] - If true, skips expressStaticGzip middleware
|
|
* @returns {ReturnType<expressStaticGzip>|ReturnType<express.static>} Express middleware function for serving static files
|
|
*/
|
|
function staticCache(staticPath, options = {}) {
|
|
const { noCache = false, skipGzipScan = false } = options;
|
|
const enableBrotli = isEnabled(process.env.ENABLE_STATIC_ASSET_BROTLI);
|
|
|
|
const setHeaders = (res, filePath) => {
|
|
if (process.env.NODE_ENV?.toLowerCase() !== 'production') {
|
|
return;
|
|
}
|
|
if (noCache) {
|
|
res.setHeader('Cache-Control', 'no-store, no-cache, must-revalidate');
|
|
return;
|
|
}
|
|
if (filePath && filePath.includes('/dist/images/')) {
|
|
return;
|
|
}
|
|
const fileName = filePath ? path.basename(filePath) : '';
|
|
|
|
if (
|
|
fileName === 'index.html' ||
|
|
fileName.endsWith('.webmanifest') ||
|
|
fileName === 'manifest.json' ||
|
|
fileName === 'sw.js' ||
|
|
fileName === 'sw-heal.js'
|
|
) {
|
|
res.setHeader('Cache-Control', 'no-store, no-cache, must-revalidate');
|
|
} else {
|
|
res.setHeader('Cache-Control', `public, max-age=${maxAge}, s-maxage=${sMaxAge}`);
|
|
}
|
|
};
|
|
|
|
if (skipGzipScan) {
|
|
return express.static(staticPath, {
|
|
setHeaders,
|
|
index: false,
|
|
});
|
|
} else {
|
|
return expressStaticGzip(staticPath, {
|
|
enableBrotli,
|
|
orderPreference: enableBrotli ? ['br', 'gz'] : ['gz'],
|
|
setHeaders,
|
|
index: false,
|
|
});
|
|
}
|
|
}
|
|
|
|
module.exports = staticCache;
|