mirror of
https://github.com/openfrontio/OpenFrontIO.git
synced 2026-06-21 11:30:43 +00:00
4aa726cfd8
## Description: Add an optional CDN_BASE env var that prefixes hashed asset URLs from asset-manifest.json, so the app can serve static assets from R2/CDN instead of the app origin. The value is determined at runtime via the EJS template (window.CDN_BASE) — empty string means "same origin," matching today's behavior. A hack to load the worker bundle: A same-origin Blob script that dynamic-import()s the cross-origin worker module and buffers early postMessage calls until the imported module's handler attaches, sidestepping the browser's refusal to construct a Worker directly from a cross-origin URL. ## Please complete the following: - [x] I have added screenshots for all UI updates - [x] I process any text displayed to the user through translateText() and I've added it to the en.json file - [x] I have added relevant tests to the test directory - [x] I confirm I have thoroughly tested these changes and take full responsibility for any bugs introduced ## Please put your Discord username so you can be contacted if a bug or regression is found: evan
153 lines
5.4 KiB
TypeScript
153 lines
5.4 KiB
TypeScript
import { describe, expect, test } from "vitest";
|
|
import { buildAssetUrl, rewriteAssetsForCdn } from "../src/core/AssetUrls";
|
|
|
|
describe("AssetUrls", () => {
|
|
test("returns hashed URLs for direct asset matches", () => {
|
|
expect(
|
|
buildAssetUrl("images/Favicon.svg", {
|
|
"images/Favicon.svg": "/_assets/images/Favicon.hash.svg",
|
|
}),
|
|
).toBe("/_assets/images/Favicon.hash.svg");
|
|
});
|
|
|
|
test("falls back to the unversioned path when manifest has no match", () => {
|
|
expect(buildAssetUrl("images/unknown.svg", {})).toBe("/images/unknown.svg");
|
|
});
|
|
|
|
test("falls back to the unversioned path for directory-like paths", () => {
|
|
const manifest = {
|
|
"maps/britanniaclassic/manifest.json":
|
|
"/_assets/maps/britanniaclassic/manifest.hash.json",
|
|
"maps/britanniaclassic/map.bin":
|
|
"/_assets/maps/britanniaclassic/map.hash.bin",
|
|
};
|
|
|
|
expect(buildAssetUrl("maps", manifest)).toBe("/maps");
|
|
expect(buildAssetUrl("maps/britanniaclassic", manifest)).toBe(
|
|
"/maps/britanniaclassic",
|
|
);
|
|
});
|
|
|
|
test("rejects dot segments in asset paths", () => {
|
|
expect(() => buildAssetUrl("../api/instance", {})).toThrow(
|
|
"Invalid asset path segment: ..",
|
|
);
|
|
expect(() => buildAssetUrl("images/%2e%2e/secret.svg", {})).toThrow(
|
|
"Invalid asset path segment: %2e%2e",
|
|
);
|
|
});
|
|
|
|
test("rejects empty asset paths", () => {
|
|
expect(() => buildAssetUrl("", {})).toThrow("Asset path must not be empty");
|
|
expect(() => buildAssetUrl("///", {})).toThrow(
|
|
"Asset path must not be empty",
|
|
);
|
|
});
|
|
|
|
test("prefixes baseUrl onto hashed URLs when provided", () => {
|
|
expect(
|
|
buildAssetUrl(
|
|
"images/Favicon.svg",
|
|
{ "images/Favicon.svg": "/_assets/images/Favicon.hash.svg" },
|
|
"https://cdn.example.com",
|
|
),
|
|
).toBe("https://cdn.example.com/_assets/images/Favicon.hash.svg");
|
|
});
|
|
|
|
test("preserves direct URL when baseUrl is empty string", () => {
|
|
expect(
|
|
buildAssetUrl(
|
|
"images/Favicon.svg",
|
|
{ "images/Favicon.svg": "/_assets/images/Favicon.hash.svg" },
|
|
"",
|
|
),
|
|
).toBe("/_assets/images/Favicon.hash.svg");
|
|
});
|
|
|
|
test("returns absolute http(s) URLs unchanged and ignores baseUrl", () => {
|
|
expect(
|
|
buildAssetUrl(
|
|
"https://example.com/foo.png",
|
|
{},
|
|
"https://cdn.example.com",
|
|
),
|
|
).toBe("https://example.com/foo.png");
|
|
expect(buildAssetUrl("HTTP://example.com/foo.png", {})).toBe(
|
|
"HTTP://example.com/foo.png",
|
|
);
|
|
});
|
|
|
|
// Manifest miss → keep same-origin; the CDN only serves what was explicitly
|
|
// hashed and uploaded, so unknown paths must not be prefixed.
|
|
test("does not prefix baseUrl on manifest misses", () => {
|
|
expect(
|
|
buildAssetUrl("images/unknown.svg", {}, "https://cdn.example.com"),
|
|
).toBe("/images/unknown.svg");
|
|
});
|
|
|
|
test("strips trailing slashes on baseUrl to avoid double slash", () => {
|
|
const manifest = {
|
|
"images/Favicon.svg": "/_assets/images/Favicon.hash.svg",
|
|
};
|
|
expect(
|
|
buildAssetUrl("images/Favicon.svg", manifest, "https://cdn.example.com/"),
|
|
).toBe("https://cdn.example.com/_assets/images/Favicon.hash.svg");
|
|
expect(
|
|
buildAssetUrl(
|
|
"images/Favicon.svg",
|
|
manifest,
|
|
"https://cdn.example.com///",
|
|
),
|
|
).toBe("https://cdn.example.com/_assets/images/Favicon.hash.svg");
|
|
});
|
|
});
|
|
|
|
describe("rewriteAssetsForCdn", () => {
|
|
test("rewrites src=/assets/ to EJS placeholder", () => {
|
|
const out = rewriteAssetsForCdn(
|
|
`<script type="module" crossorigin src="/assets/index-XXX.js"></script>`,
|
|
);
|
|
expect(out).toBe(
|
|
`<script type="module" crossorigin src="<%- locals.cdnBaseRaw || "" %>/assets/index-XXX.js"></script>`,
|
|
);
|
|
});
|
|
|
|
test("rewrites href=/assets/ for modulepreload and stylesheet links", () => {
|
|
const out = rewriteAssetsForCdn(
|
|
`<link rel="modulepreload" href="/assets/vendor-XXX.js">\n<link rel="stylesheet" href="/assets/index-XXX.css">`,
|
|
);
|
|
expect(out).toBe(
|
|
`<link rel="modulepreload" href="<%- locals.cdnBaseRaw || "" %>/assets/vendor-XXX.js">\n<link rel="stylesheet" href="<%- locals.cdnBaseRaw || "" %>/assets/index-XXX.css">`,
|
|
);
|
|
});
|
|
|
|
test("supports single-quoted attribute values", () => {
|
|
expect(rewriteAssetsForCdn(`<script src='/assets/x.js'></script>`)).toBe(
|
|
`<script src='<%- locals.cdnBaseRaw || "" %>/assets/x.js'></script>`,
|
|
);
|
|
});
|
|
|
|
test("does not rewrite /_assets/ (underscore manifest paths)", () => {
|
|
const html = `<link rel="icon" href="/_assets/images/Favicon.hash.svg">`;
|
|
expect(rewriteAssetsForCdn(html)).toBe(html);
|
|
});
|
|
|
|
test("does not rewrite already-absolute asset URLs", () => {
|
|
const html = `<script src="https://example.com/assets/foo.js"></script>`;
|
|
expect(rewriteAssetsForCdn(html)).toBe(html);
|
|
});
|
|
|
|
// Inline scripts containing the literal "/assets/..." string must survive
|
|
// unrewrite — the regex requires whitespace before src=/href=, and inside a
|
|
// JS string literal there's no preceding `src=`/`href=` token at all.
|
|
test("does not mangle /assets/ inside inline script string literals", () => {
|
|
const html = `<script>const url = "/assets/foo";</script>`;
|
|
expect(rewriteAssetsForCdn(html)).toBe(html);
|
|
});
|
|
|
|
test("does not match data-src or other custom attributes", () => {
|
|
const html = `<img data-src="/assets/foo.png">`;
|
|
expect(rewriteAssetsForCdn(html)).toBe(html);
|
|
});
|
|
});
|