ghost
Version:
The professional publishing platform
81 lines (66 loc) • 2.34 kB
JavaScript
const path = require('path');
const config = require('../../../shared/config');
const themeEngine = require('../../services/theme-engine');
const express = require('../../../shared/express');
function isDeniedFile(file) {
const deniedFileTypes = ['.hbs', '.md', '.json', '.lock', '.log'];
const deniedFiles = ['gulpfile.js', 'gruntfile.js'];
const ext = path.extname(file);
const base = path.basename(file);
return deniedFiles.includes(base) || deniedFileTypes.includes(ext);
}
/**
* Copy from:
* https://github.com/pillarjs/send/blob/b69cbb3dc4c09c37917d08a4c13fcd1bac97ade5/index.js#L987-L1003
*
* Allows V8 to only deoptimize this fn instead of all
* of send().
*
* @param {string} filePath
* @returns {string|number} returns -1 number if decode decodeURIComponent throws
*/
function decode(filePath) {
try {
return decodeURIComponent(filePath);
} catch (err) {
return -1;
}
}
/**
*
* @param {string} file path to a requested file
* @returns {boolean}
*/
function isAllowedFile(file) {
const decodedFilePath = decode(file);
if (decodedFilePath === -1) {
return false;
}
const normalizedFilePath = path.normalize(decodedFilePath);
const allowedFiles = ['manifest.json', 'assetlinks.json'];
const allowedPath = '/assets/';
const alwaysDeny = ['.hbs'];
const ext = path.extname(normalizedFilePath);
const base = path.basename(normalizedFilePath);
return allowedFiles.includes(base) || (normalizedFilePath.startsWith(allowedPath) && !alwaysDeny.includes(ext));
}
function forwardToExpressStatic(req, res, next) {
if (!themeEngine.getActive()) {
return next();
}
express.static(themeEngine.getActive().path, {
// @NOTE: the maxAge config passed below are in milliseconds and the config
// is specified in seconds. See https://github.com/expressjs/serve-static/issues/150 for more context
maxAge: config.get('caching:theme:maxAge') * 1000
}
)(req, res, next);
}
function staticTheme() {
return function denyStatic(req, res, next) {
if (!isAllowedFile(req.path.toLowerCase()) && isDeniedFile(req.path.toLowerCase())) {
return next();
}
return forwardToExpressStatic(req, res, next);
};
}
module.exports = staticTheme;