166 lines
5.8 KiB
JavaScript
166 lines
5.8 KiB
JavaScript
#!/usr/bin/env node
|
|
/**
|
|
* prerender-seo.mjs — Generates route-specific HTML for SEO-important pages.
|
|
*
|
|
* How it works:
|
|
* 1. Starts a Vite dev server in SSR mode (no HTTP server, just the transform pipeline)
|
|
* 2. Loads src/entry-server.tsx through Vite — handles TSX, CSS, SVG, path aliases
|
|
* 3. For each route with seoMeta in seoRoutes.ts:
|
|
* a. Renders the React component to HTML via renderToString
|
|
* b. Replaces <title>, <meta>, OG/Twitter tags, canonical URL from seoMeta
|
|
* c. Injects the rendered HTML into #root-seo
|
|
* 4. Writes to dist/{route}/index.html
|
|
*
|
|
* nginx's try_files ($uri/) serves these automatically.
|
|
*
|
|
* Run after `vite build`: node scripts/prerender-seo.mjs
|
|
*/
|
|
|
|
import { readFileSync, writeFileSync, mkdirSync, existsSync } from 'fs';
|
|
import { dirname, join } from 'path';
|
|
import { fileURLToPath } from 'url';
|
|
import { createServer } from 'vite';
|
|
|
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
const distDir = join(__dirname, '..', 'dist');
|
|
|
|
// Check that dist exists (vite build must have run first)
|
|
if (!existsSync(join(distDir, 'index.html'))) {
|
|
console.error('❌ dist/index.html not found. Run `vite build` first.');
|
|
process.exit(1);
|
|
}
|
|
|
|
const baseHtml = readFileSync(join(distDir, 'index.html'), 'utf-8');
|
|
const DOMAIN = 'https://velxio.dev';
|
|
|
|
// ── Mock browser globals for SSR ────────────────────────────────────────────
|
|
// Zustand's persist middleware and some components access these at import time.
|
|
if (typeof globalThis.localStorage === 'undefined') {
|
|
globalThis.localStorage = {
|
|
getItem: () => null,
|
|
setItem: () => {},
|
|
removeItem: () => {},
|
|
clear: () => {},
|
|
length: 0,
|
|
key: () => null,
|
|
};
|
|
}
|
|
if (typeof globalThis.window === 'undefined') {
|
|
globalThis.window = globalThis;
|
|
}
|
|
if (typeof globalThis.navigator === 'undefined') {
|
|
globalThis.navigator = { userAgent: 'prerender-seo' };
|
|
}
|
|
if (typeof globalThis.matchMedia === 'undefined') {
|
|
globalThis.matchMedia = () => ({ matches: false, addListener: () => {}, removeListener: () => {} });
|
|
}
|
|
|
|
// ── Start Vite in SSR mode ──────────────────────────────────────────────────
|
|
console.log('🔧 Starting Vite SSR transform pipeline...');
|
|
|
|
const vite = await createServer({
|
|
server: { middlewareMode: true },
|
|
appType: 'custom',
|
|
logLevel: 'warn',
|
|
});
|
|
|
|
let generated = 0;
|
|
|
|
try {
|
|
// Load entry-server.tsx through Vite's transform pipeline
|
|
const { getPrerenderedRoutes, render } = await vite.ssrLoadModule('/src/entry-server.tsx');
|
|
const routes = getPrerenderedRoutes();
|
|
|
|
console.log(`📄 Prerendering ${routes.length} SEO pages...\n`);
|
|
|
|
for (const route of routes) {
|
|
const { seoMeta } = route;
|
|
if (!seoMeta) continue;
|
|
|
|
// Render the React component to HTML
|
|
let bodyHtml = render(route.path);
|
|
|
|
// Build the page HTML
|
|
let html = baseHtml;
|
|
|
|
// Replace <title>
|
|
html = html.replace(/<title>[^<]*<\/title>/, `<title>${seoMeta.title}</title>`);
|
|
|
|
// Replace meta description
|
|
html = html.replace(
|
|
/<meta name="description" content="[^"]*"/,
|
|
`<meta name="description" content="${seoMeta.description}"`
|
|
);
|
|
|
|
// Add/replace canonical URL
|
|
const canonicalTag = `<link rel="canonical" href="${seoMeta.url}" />`;
|
|
if (html.includes('<link rel="canonical"')) {
|
|
html = html.replace(/<link rel="canonical"[^>]*\/>/, canonicalTag);
|
|
} else {
|
|
html = html.replace('</head>', ` ${canonicalTag}\n </head>`);
|
|
}
|
|
|
|
// Replace OG tags
|
|
html = html.replace(
|
|
/<meta property="og:title" content="[^"]*"/,
|
|
`<meta property="og:title" content="${seoMeta.title}"`
|
|
);
|
|
html = html.replace(
|
|
/<meta property="og:description" content="[^"]*"/,
|
|
`<meta property="og:description" content="${seoMeta.description}"`
|
|
);
|
|
html = html.replace(
|
|
/<meta property="og:url" content="[^"]*"/,
|
|
`<meta property="og:url" content="${seoMeta.url}"`
|
|
);
|
|
|
|
// Replace Twitter tags
|
|
html = html.replace(
|
|
/<meta name="twitter:title" content="[^"]*"/,
|
|
`<meta name="twitter:title" content="${seoMeta.title}"`
|
|
);
|
|
html = html.replace(
|
|
/<meta name="twitter:description" content="[^"]*"/,
|
|
`<meta name="twitter:description" content="${seoMeta.description}"`
|
|
);
|
|
|
|
// Replace #root-seo content with SSR-rendered body (or fallback to title+description)
|
|
const seoBody = bodyHtml
|
|
? bodyHtml
|
|
: `<h1>${seoMeta.title.split(' | ')[0]}</h1><p>${seoMeta.description}</p>`;
|
|
|
|
html = html.replace(
|
|
/<div id="root-seo"[^>]*>[\s\S]*?<\/div>\s*<script/,
|
|
`<div id="root-seo" aria-hidden="true">${seoBody}</div>\n <script`
|
|
);
|
|
|
|
// Write to dist/{path}/index.html
|
|
const routePath = route.path === '/' ? '' : route.path.slice(1);
|
|
if (routePath) {
|
|
const dir = join(distDir, routePath);
|
|
if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
|
|
writeFileSync(join(dir, 'index.html'), html, 'utf-8');
|
|
} else {
|
|
// Root — update dist/index.html in place
|
|
writeFileSync(join(distDir, 'index.html'), html, 'utf-8');
|
|
}
|
|
|
|
generated++;
|
|
const ssrStatus = bodyHtml ? '✓' : '⚠ (meta only)';
|
|
console.log(` ${ssrStatus} ${route.path}`);
|
|
}
|
|
|
|
// Also ensure root index.html has a canonical tag
|
|
let rootHtml = readFileSync(join(distDir, 'index.html'), 'utf-8');
|
|
const rootCanonical = `<link rel="canonical" href="${DOMAIN}/" />`;
|
|
if (!rootHtml.includes('<link rel="canonical"')) {
|
|
rootHtml = rootHtml.replace('</head>', ` ${rootCanonical}\n </head>`);
|
|
writeFileSync(join(distDir, 'index.html'), rootHtml, 'utf-8');
|
|
}
|
|
|
|
} finally {
|
|
await vite.close();
|
|
}
|
|
|
|
console.log(`\n✅ Prerendered ${generated} SEO pages with SSR content`);
|