// Copyright (c) 2026 Cesanta Software Limited
// All rights reserved
//
// Generate a Mongoose packed filesystem C file from an HTML file after
// inlining referenced CSS, JS, images, and icons.
//
// Usage:
// node html2c.js [-o OUTPUT.c] HTML_FILE [HTML_FILE ...]
const fs = require('fs');
const path = require('path');
const http = require('http');
const https = require('https');
const zlib = require('zlib');
const args = process.argv.slice(2);
let outputFile = null;
const inputFiles = [];
const useMap = new Map();
for (let i = 0; i < args.length; i++) {
if (args[i] === '-o' && i + 1 < args.length) {
outputFile = args[++i];
} else if (args[i] === '--use' && i + 1 < args.length) {
const eq = args[++i].indexOf('=');
if (eq > 0) useMap.set(args[i].slice(0, eq), args[i].slice(eq + 1));
} else if (!args[i].startsWith('-')) {
inputFiles.push(args[i]);
}
}
if (inputFiles.length === 0) {
console.error('Usage: node html2c.js [-o OUTPUT.c] HTML_FILE [HTML_FILE ...]');
process.exit(1);
}
function fetchUrl(url) {
return new Promise((resolve, reject) => {
const client = url.startsWith('https') ? https : http;
client
.get(url, res => {
const chunks = [];
res.on('data', chunk => chunks.push(chunk));
res.on('end', () => resolve(Buffer.concat(chunks)));
res.on('error', reject);
})
.on('error', reject);
});
}
async function readAsset(name, encoding, baseDir) {
const override = useMap.get(name);
if (override != null) name = override;
if (name.startsWith('http://') || name.startsWith('https://')) {
const data = await fetchUrl(name);
return encoding ? data.toString(encoding) : data;
}
const fullPath = path.isAbsolute(name) ? name : path.join(baseDir, name);
return fs.readFileSync(fullPath, encoding || null);
}
function attr(match, name) {
const re = new RegExp(`${name}=["']([^"']+)["']`, 'i');
const m = match.match(re);
return m ? m[1] : null;
}
function replaceAttr(match, name, value) {
const re = new RegExp(`${name}=(["'])[^"']+\\1`, 'i');
return match.replace(re, `${name}="${value}"`);
}
async function inlineCss(match, baseDir) {
const href = attr(match, 'href');
if (!href) return match;
const css = await readAsset(href, 'utf8', baseDir);
return ``;
}
async function inlineJs(match, baseDir) {
const src = attr(match, 'src');
if (!src) return match;
const js = await readAsset(src, 'utf8', baseDir);
return ``;
}
async function inlineImg(match, baseDir) {
const src = attr(match, 'src');
if (!src) return match;
const ext = path.extname(src).toLowerCase();
let mime = 'image/png';
if (ext === '.svg') mime = 'image/svg+xml';
else if (ext === '.jpg' || ext === '.jpeg') mime = 'image/jpeg';
else if (ext === '.gif') mime = 'image/gif';
else if (ext === '.webp') mime = 'image/webp';
else if (ext === '.ico') mime = 'image/x-icon';
const data = await readAsset(src, null, baseDir);
if (ext === '.svg') {
const svg = Buffer.isBuffer(data) ? data.toString('utf8') : data;
return replaceAttr(
match,
'src',
`data:${mime};utf8,${encodeURIComponent(svg)}`,
);
}
return replaceAttr(
match,
'src',
`data:${mime};base64,${Buffer.from(data).toString('base64')}`,
);
}
async function inlineSvg(match, baseDir) {
const href = attr(match, 'href');
if (!href || !href.toLowerCase().endsWith('.svg')) return match;
const data = await readAsset(href, null, baseDir);
return replaceAttr(
match,
'href',
`data:image/svg+xml;base64,${Buffer.from(data).toString('base64')}`,
);
}
async function processHtml(content, baseDir) {
let result = content;
const cssRegex = /]+rel=["']stylesheet["'][^>]*>/g;
const jsRegex = /