const fs = require("fs") const path = require("path") const etag = require("etag") const babel = require("babel-core") const unpkgRewrite = require("babel-plugin-unpkg-rewrite") const getMetadata = require("./utils/getMetadata") const getFileContentType = require("./utils/getFileContentType") const getIndexHTML = require("./utils/getIndexHTML") /** * Automatically generate HTML pages that show package contents. */ const AutoIndex = !process.env.DISABLE_INDEX /** * Maximum recursion depth for meta listings. */ const MaximumDepth = 128 const FileTransforms = { expand: function(file, dependencies, callback) { const options = { plugins: [unpkgRewrite(dependencies)] } babel.transformFile(file, options, function(error, result) { callback(error, result && result.code) }) } } /** * Send the file, JSON metadata, or HTML directory listing. */ function serveFile(req, res, next) { if (req.query.meta != null) { // Serve JSON metadata. getMetadata(req.packageDir, req.filename, req.stats, MaximumDepth, function(error, metadata) { if (error) { console.error(error) res .status(500) .type("text") .send(`Cannot generate metadata for ${req.packageSpec}${req.filename}`) } else { // Cache metadata for 1 year. res .set({ "Cache-Control": "public, max-age=31536000", "Cache-Tag": "meta" }) .send(metadata) } }) } else if (req.stats.isFile()) { // Serve a file. const file = path.join(req.packageDir, req.filename) let contentType = getFileContentType(file) if (contentType === "text/html") contentType = "text/plain" // We can't serve HTML because bad people :( if (contentType === "application/javascript" && req.query.module != null) { // Serve a JavaScript module. const dependencies = Object.assign( {}, req.packageConfig.peerDependencies, req.packageConfig.dependencies ) FileTransforms.expand(file, dependencies, function(error, code) { if (error) { console.error(error) const debugInfo = error.constructor.name + ": " + error.message.replace(/^.*?\/unpkg-.+?\//, `/${req.packageSpec}/`) + "\n\n" + error.codeFrame res .status(500) .type("text") .send(`Cannot generate module for ${req.packageSpec}${req.filename}\n\n${debugInfo}`) } else { // Cache modules for 1 year. res .set({ "Content-Type": contentType, "Content-Length": Buffer.byteLength(code), "Cache-Control": "public, max-age=31536000", "Cache-Tag": "file,js-file,js-module" }) .send(code) } }) } else { // Serve some other static file. const tags = ["file"] const ext = path.extname(req.filename).substr(1) if (ext) tags.push(`${ext}-file`) // Cache files for 1 year. res.set({ "Content-Type": contentType, "Content-Length": req.stats.size, "Cache-Control": "public, max-age=31536000", "Last-Modified": req.stats.mtime.toUTCString(), ETag: etag(req.stats), "Cache-Tag": tags.join(",") }) const stream = fs.createReadStream(file) stream.on("error", function(error) { console.error(`Cannot send file ${req.packageSpec}${req.filename}`) console.error(error) res.sendStatus(500) }) stream.pipe(res) } } else if (AutoIndex && req.stats.isDirectory()) { // Serve an HTML directory listing. getIndexHTML(req.packageInfo, req.packageVersion, req.packageDir, req.filename, function( error, html ) { if (error) { console.error(error) res .status(500) .type("text") .send(`Cannot generate index page for ${req.packageSpec}${req.filename}`) } else { // Cache HTML directory listings for 1 minute. res .set({ "Cache-Control": "public, max-age=60", "Cache-Tag": "index" }) .send(html) } }) } else { res .status(403) .type("text") .send(`Cannot serve ${req.packageSpec}${req.filename}; it's not a file`) } } module.exports = serveFile