cache ArcGIS basemap tiles on disk for faster map loads

This commit is contained in:
mrkmntal 2026-04-07 22:11:03 -04:00
commit 437100c433
5 changed files with 89 additions and 9 deletions

View file

@ -5,7 +5,15 @@ import { readFile } from 'fs/promises';
import { exec } from 'child_process';
import { promisify } from 'util';
import {
weatherProxy, radarProxy, outlookProxy, mesonetProxy, forecastProxy, openMeteoProxy, rainViewerProxy,
weatherProxy,
radarProxy,
outlookProxy,
mesonetProxy,
forecastProxy,
openMeteoProxy,
rainViewerProxy,
arcGisServerProxy,
arcGisServicesProxy,
} from './proxy/handlers.mjs';
import playlist from './src/playlist.mjs';
import OVERRIDES from './src/overrides.mjs';
@ -256,6 +264,8 @@ if (!process.env?.STATIC) {
app.use('/forecast/', forecastProxy);
app.use('/open-meteo/', openMeteoProxy);
app.use('/rainviewer/', rainViewerProxy);
app.use('/arcgis-server/', arcGisServerProxy);
app.use('/arcgis-services/', arcGisServicesProxy);
// Playlist route is available in server mode (not in static mode)
app.get('/playlist.json', playlist);

View file

@ -32,10 +32,18 @@ const STALE_TIME_LIMIT_MS = 3 * 60 * 60 * 1000;
const PERSISTED_HOSTS = new Set([
'api.open-meteo.com',
'api.rainviewer.com',
'server.arcgisonline.com',
'services.arcgisonline.com',
]);
const BINARY_PERSISTED_HOSTS = new Set([
'server.arcgisonline.com',
'services.arcgisonline.com',
]);
const HOST_FALLBACK_TTLS = {
'api.open-meteo.com': 10 * 60,
'api.rainviewer.com': 2 * 60,
'server.arcgisonline.com': 7 * 24 * 60 * 60,
'services.arcgisonline.com': 7 * 24 * 60 * 60,
};
class HttpCache {
@ -55,16 +63,30 @@ class HttpCache {
return path.join(CACHE_DIR, `${HttpCache.hashKey(key)}.json`);
}
static shouldPersist(url, options, response) {
if (options?.encoding === 'binary') return false;
if (typeof response?.data !== 'string') return false;
static getCacheBodyFilePath(key) {
return path.join(CACHE_DIR, `${HttpCache.hashKey(key)}.bin`);
}
static getPersistedHost(url) {
try {
const parsedUrl = new URL(url);
return PERSISTED_HOSTS.has(parsedUrl.hostname);
return parsedUrl.hostname;
} catch {
return null;
}
}
static shouldPersist(url, options, response) {
const host = HttpCache.getPersistedHost(url);
if (!host || !PERSISTED_HOSTS.has(host)) {
return false;
}
if (options?.encoding === 'binary') {
return BINARY_PERSISTED_HOSTS.has(host) && Buffer.isBuffer(response?.data);
}
return typeof response?.data === 'string';
}
static getHostFallbackTtl(url) {
@ -96,10 +118,17 @@ class HttpCache {
}
if (now > parsed.entry.expiry + STALE_TIME_LIMIT_MS) {
if (parsed.entry.binaryBody === true) {
await unlink(HttpCache.getCacheBodyFilePath(parsed.key)).catch(() => null);
}
await unlink(filePath);
return;
}
if (parsed.entry.binaryBody === true) {
parsed.entry.data = await readFile(HttpCache.getCacheBodyFilePath(parsed.key));
}
this.cache.set(parsed.key, parsed.entry);
}));
} catch (error) {
@ -110,6 +139,18 @@ class HttpCache {
static async persistEntry(key, entry) {
try {
await mkdir(CACHE_DIR, { recursive: true });
if (Buffer.isBuffer(entry.data)) {
const metadata = {
...entry,
data: undefined,
binaryBody: true,
};
await writeFile(HttpCache.getCacheBodyFilePath(key), entry.data);
await writeFile(HttpCache.getCacheFilePath(key), JSON.stringify({ key, entry: metadata }));
return;
}
await writeFile(HttpCache.getCacheFilePath(key), JSON.stringify({ key, entry }));
} catch (error) {
console.warn(`⚠️ Cache save | Failed to persist cache entry ${key}: ${error.message}`);
@ -118,6 +159,7 @@ class HttpCache {
static async deletePersistedEntry(key) {
try {
await unlink(HttpCache.getCacheBodyFilePath(key)).catch(() => null);
await unlink(HttpCache.getCacheFilePath(key));
} catch (error) {
if (error.code !== 'ENOENT') {

View file

@ -64,3 +64,19 @@ export const rainViewerProxy = async (req, res) => {
skipParams: ['u'],
});
};
export const arcGisServerProxy = async (req, res) => {
await cache.handleRequest(req, res, 'https://server.arcgisonline.com', {
serviceName: 'ArcGIS Server',
skipParams: ['u'],
encoding: 'binary',
});
};
export const arcGisServicesProxy = async (req, res) => {
await cache.handleRequest(req, res, 'https://services.arcgisonline.com', {
serviceName: 'ArcGIS Services',
skipParams: ['u'],
encoding: 'binary',
});
};

View file

@ -4,8 +4,12 @@ import { getSmallIconFromWmoCode } from '../icons.mjs';
import { getOpenMeteoObservationSnapshot } from './weather.mjs';
import { temperature } from './units.mjs';
const BASE_MAP_URL = 'https://server.arcgisonline.com/ArcGIS/rest/services/Canvas/World_Light_Gray_Base/MapServer/tile/{z}/{y}/{x}';
const BOUNDARY_MAP_URL = 'https://services.arcgisonline.com/ArcGIS/rest/services/Reference/World_Boundaries_and_Places/MapServer/tile/{z}/{y}/{x}';
const getBaseMapUrl = () => (window.WS4KP_SERVER_AVAILABLE
? '/arcgis-server/ArcGIS/rest/services/Canvas/World_Light_Gray_Base/MapServer/tile/{z}/{y}/{x}'
: 'https://server.arcgisonline.com/ArcGIS/rest/services/Canvas/World_Light_Gray_Base/MapServer/tile/{z}/{y}/{x}');
const getBoundaryMapUrl = () => (window.WS4KP_SERVER_AVAILABLE
? '/arcgis-services/ArcGIS/rest/services/Reference/World_Boundaries_and_Places/MapServer/tile/{z}/{y}/{x}'
: 'https://services.arcgisonline.com/ArcGIS/rest/services/Reference/World_Boundaries_and_Places/MapServer/tile/{z}/{y}/{x}');
const DEFAULT_MAX_NEARBY_MARKERS = 7;
const MIN_CITY_DISTANCE_METERS = 25000;
const MIN_MARKER_PIXEL_DISTANCE = 85;
@ -26,14 +30,14 @@ const createMap = (mapElement) => window.L.map(mapElement, {
});
const addBaseLayers = (map) => {
const baseLayer = window.L.tileLayer(BASE_MAP_URL, {
const baseLayer = window.L.tileLayer(getBaseMapUrl(), {
maxZoom: 10,
minZoom: 1,
crossOrigin: true,
className: 'radar-base-layer',
}).addTo(map);
const boundaryLayer = window.L.tileLayer(BOUNDARY_MAP_URL, {
const boundaryLayer = window.L.tileLayer(getBoundaryMapUrl(), {
maxZoom: 10,
minZoom: 1,
opacity: 0.6,

View file

@ -50,6 +50,14 @@ const rewriteUrl = (_url) => {
url.protocol = window.location.protocol;
url.host = window.location.host;
url.pathname = `/rainviewer${url.pathname}`;
} else if (url.origin === 'https://server.arcgisonline.com') {
url.protocol = window.location.protocol;
url.host = window.location.host;
url.pathname = `/arcgis-server${url.pathname}`;
} else if (url.origin === 'https://services.arcgisonline.com') {
url.protocol = window.location.protocol;
url.host = window.location.host;
url.pathname = `/arcgis-services${url.pathname}`;
} else if (typeof OVERRIDES !== 'undefined' && OVERRIDES?.RADAR_HOST && url.origin === `https://${OVERRIDES.RADAR_HOST}`) {
// Handle override radar host
url.protocol = window.location.protocol;