Files
eranos/vite.config.ts
T
Alex Gleason 22f13c1505 Replace __DITTO_CONFIG__ global with import.meta.env.DITTO_CONFIG and remove ThemeSchemaCompat
Move build-time ditto.json injection from a Vite define global to
import.meta.env.DITTO_CONFIG (a JSON string parsed and validated at
runtime via DittoConfigSchema). Remove the global type declaration
from vite-env.d.ts.

Drop ThemeSchemaCompat and its legacy "black"/"pink" migration code
from AppProvider and NostrSync — invalid theme values now simply fail
Zod validation.

Fix a latent bug where a partial feedSettings from ditto.json would
replace the full hardcoded defaults; defaultConfig now deep-merges
feedSettings.
2026-04-01 23:16:33 -05:00

178 lines
5.2 KiB
TypeScript

import process from "node:process";
import { execSync } from "node:child_process";
import { createRequire } from "node:module";
import fs from "node:fs";
import path from "node:path";
import react from "@vitejs/plugin-react";
import { visualizer } from "rollup-plugin-visualizer";
import { defineConfig, type Plugin } from "vite";
import { DittoConfigSchema } from "./src/lib/schemas";
/**
* Load and validate the build-time ditto.json configuration file.
* Returns the parsed config object, or `undefined` if the file doesn't exist.
* Set the CONFIG_FILE env var to override the default path ("./ditto.json").
*/
function loadDittoConfig(): object | undefined {
const configPath = path.resolve(process.env.CONFIG_FILE ?? "./ditto.json");
let raw: string;
try {
raw = fs.readFileSync(configPath, "utf-8");
} catch {
// File not found — no build-time config
return undefined;
}
const json = JSON.parse(raw);
const result = DittoConfigSchema.parse(json);
return result;
}
/**
* Copy all files from `src` into `dest`, overwriting existing files.
* Recursively handles subdirectories.
*/
function copyDirSync(src: string, dest: string): void {
if (!fs.existsSync(src)) return;
fs.mkdirSync(dest, { recursive: true });
for (const entry of fs.readdirSync(src, { withFileTypes: true })) {
const srcPath = path.join(src, entry.name);
const destPath = path.join(dest, entry.name);
if (entry.isDirectory()) {
copyDirSync(srcPath, destPath);
} else {
fs.copyFileSync(srcPath, destPath);
}
}
}
/**
* Vite plugin that merges an external public directory on top of the default one.
* Set the PUBLIC_DIR env var to a directory path. Files in that directory take
* precedence over files in the built-in `public/` directory.
*
* - In build mode, files are copied into the output after the default public dir.
* - In dev mode, the external directory is served with higher priority.
*/
function mergePublicDir(externalDir: string): Plugin {
const resolved = path.resolve(externalDir);
return {
name: "ditto:merge-public-dir",
configureServer(server) {
// Serve files from the external public dir before the default public dir.
server.middlewares.use((req, res, next) => {
if (!req.url) return next();
const urlPath = decodeURIComponent(new URL(req.url, "http://localhost").pathname);
const filePath = path.join(resolved, urlPath);
try {
const stat = fs.statSync(filePath);
if (stat.isFile()) {
// Let Vite's static middleware handle it by pointing to the file.
const stream = fs.createReadStream(filePath);
stream.pipe(res);
return;
}
} catch {
// File not found in external dir — fall through to default public dir
}
next();
});
},
writeBundle(options) {
const outDir = options.dir ?? path.resolve("dist");
copyDirSync(resolved, outDir);
},
};
}
const dittoConfig = loadDittoConfig();
const publicDir = process.env.PUBLIC_DIR;
const require = createRequire(import.meta.url);
const pkg = require("./package.json") as { version: string };
/** Short commit SHA — prefer CI env var, fall back to git. */
function getCommitSha(): string {
if (process.env.CI_COMMIT_SHORT_SHA) return process.env.CI_COMMIT_SHORT_SHA;
try {
return execSync("git rev-parse --short HEAD", { encoding: "utf-8" }).trim();
} catch {
return "";
}
}
/** Git tag for the current commit — prefer CI env var, fall back to git. Empty string if untagged. */
function getCommitTag(): string {
if (process.env.CI_COMMIT_TAG) return process.env.CI_COMMIT_TAG;
try {
return execSync("git describe --exact-match --tags HEAD 2>/dev/null", { encoding: "utf-8" }).trim();
} catch {
return "";
}
}
// https://vitejs.dev/config/
export default defineConfig(() => {
return {
server: {
host: "::",
port: 8080,
},
plugins: [
react(),
visualizer({
filename: "dist/bundle.html",
template: "treemap",
gzipSize: true,
}),
...(publicDir ? [mergePublicDir(publicDir)] : []),
],
define: {
'import.meta.env.DITTO_CONFIG': JSON.stringify(JSON.stringify(dittoConfig ?? null)),
'import.meta.env.VERSION': JSON.stringify(pkg.version),
'import.meta.env.BUILD_DATE': JSON.stringify(new Date().toISOString()),
'import.meta.env.COMMIT_SHA': JSON.stringify(getCommitSha()),
'import.meta.env.COMMIT_TAG': JSON.stringify(getCommitTag()),
},
test: {
globals: true,
environment: 'jsdom',
setupFiles: './src/test/setup.ts',
onConsoleLog(log) {
return !log.includes("React Router Future Flag Warning");
},
env: {
DEBUG_PRINT_LIMIT: '0', // Suppress DOM output that exceeds AI context windows
},
},
build: {
target: 'esnext',
rollupOptions: {
output: {
manualChunks(id) {
// Consolidate lucide icons into a single chunk instead of 60+ micro-chunks.
if (id.includes('node_modules/lucide-react')) {
return 'lucide-icons';
}
},
},
},
},
optimizeDeps: {
exclude: ['@capacitor/filesystem', '@capacitor/share'],
},
resolve: {
alias: {
"@": path.resolve(__dirname, "./src"),
},
},
};
});