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