Add ?module support for HTML files

Closes #113
This commit is contained in:
Michael Jackson
2018-09-04 15:58:52 -07:00
parent 311a1ffa7e
commit 21ed6ee42e
7 changed files with 187 additions and 31 deletions

View File

@ -1,4 +1,5 @@
const serveAutoIndexPage = require("./serveAutoIndexPage");
const serveHTMLModule = require("./serveHTMLModule");
const serveJavaScriptModule = require("./serveJavaScriptModule");
const serveStaticFile = require("./serveStaticFile");
const serveMetadata = require("./serveMetadata");
@ -16,7 +17,18 @@ function serveFile(req, res) {
}
if (req.query.module != null) {
return serveJavaScriptModule(req, res);
if (req.entry.contentType === "application/javascript") {
return serveJavaScriptModule(req, res);
}
if (req.entry.contentType === "text/html") {
return serveHTMLModule(req, res);
}
return res
.status(403)
.type("text")
.send("?module mode is available only for JavaScript and HTML files");
}
serveStaticFile(req, res);

View File

@ -0,0 +1,50 @@
const etag = require("etag");
const cheerio = require("cheerio");
const getContentTypeHeader = require("../utils/getContentTypeHeader");
const rewriteBareModuleIdentifiers = require("../utils/rewriteBareModuleIdentifiers");
function serveHTMLModule(req, res) {
try {
const $ = cheerio.load(req.entry.content.toString("utf8"));
$("script[type=module]").each((index, element) => {
$(element).html(
rewriteBareModuleIdentifiers($(element).html(), req.packageConfig)
);
});
const code = $.html();
res
.set({
"Content-Length": Buffer.byteLength(code),
"Content-Type": getContentTypeHeader(req.entry.contentType),
"Cache-Control": "public, max-age=31536000, immutable", // 1 year
ETag: etag(code),
"Cache-Tag": "file, html-file, html-module"
})
.send(code);
} catch (error) {
console.error(error);
const errorName = error.constructor.name;
const errorMessage = error.message.replace(
/^.*?\/unpkg-.+?\//,
`/${req.packageSpec}/`
);
const codeFrame = error.codeFrame;
const debugInfo = `${errorName}: ${errorMessage}\n\n${codeFrame}`;
res
.status(500)
.type("text")
.send(
`Cannot generate module for ${req.packageSpec}${
req.filename
}\n\n${debugInfo}`
);
}
}
module.exports = serveHTMLModule;

View File

@ -1,35 +1,9 @@
const etag = require("etag");
const babel = require("babel-core");
const getContentTypeHeader = require("../utils/getContentTypeHeader");
const unpkgRewrite = require("../plugins/unpkgRewrite");
function rewriteBareModuleIdentifiers(code, packageConfig) {
const dependencies = Object.assign(
{},
packageConfig.peerDependencies,
packageConfig.dependencies
);
const options = {
// Ignore .babelrc and package.json babel config
// because we haven't installed dependencies so
// we can't load plugins; see #84
babelrc: false,
plugins: [unpkgRewrite(dependencies)]
};
return babel.transform(code, options).code;
}
const rewriteBareModuleIdentifiers = require("../utils/rewriteBareModuleIdentifiers");
function serveJavaScriptModule(req, res) {
if (req.entry.contentType !== "application/javascript") {
return res
.status(403)
.type("text")
.send("?module mode is available only for JavaScript files");
}
try {
const code = rewriteBareModuleIdentifiers(
req.entry.content.toString("utf8"),