mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-04 17:04:59 +03:00
6e6a4822f2
- Currently it's assumed that public files are 100% static - With card assets, we're using it for files that are partially static, but can change between reboots and theme changes - We already have a system for managing cache busting across theme changes and restarts - the ?v= key that is added via the asset helper - This was already in place and used, but servePublicFile's internal cache didn't honor this key, and cached for the lifetime of boot - This small change means that if a ?v= query param is present on a request for a public file, that we pay attention to it. Else we cache as before
119 lines
4.1 KiB
JavaScript
119 lines
4.1 KiB
JavaScript
const crypto = require('crypto');
|
|
const fs = require('fs-extra');
|
|
const path = require('path');
|
|
const errors = require('@tryghost/errors');
|
|
const config = require('../../../shared/config');
|
|
const urlUtils = require('../../../shared/url-utils');
|
|
const tpl = require('@tryghost/tpl');
|
|
|
|
const messages = {
|
|
imageNotFound: 'Image not found',
|
|
fileNotFound: 'File not found'
|
|
};
|
|
|
|
/**
|
|
* If this request has a ?v= param, make sure the cache has the same key
|
|
*
|
|
* @param {Object} req
|
|
* @param {Object} cache
|
|
* @returns {boolean}
|
|
*/
|
|
function matchCacheKey(req, cache) {
|
|
if (req.query && req.query.v && cache && cache.key) {
|
|
return req.query.v === cache.key;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function createPublicFileMiddleware(location, file, mime, maxAge) {
|
|
let cache;
|
|
// These files are provided by Ghost, and therefore live inside of the core folder
|
|
const staticFilePath = config.get('paths').publicFilePath;
|
|
// These files are built on the fly, and must be saved in the content folder
|
|
const builtFilePath = config.getContentPath('public');
|
|
|
|
let locationPath = location === 'static' ? staticFilePath : builtFilePath;
|
|
|
|
const filePath = file.match(/^public/) ? path.join(locationPath, file.replace(/^public/, '')) : path.join(locationPath, file);
|
|
const blogRegex = /(\{\{blog-url\}\})/g;
|
|
|
|
return function servePublicFileMiddleware(req, res, next) {
|
|
if (cache && matchCacheKey(req, cache)) {
|
|
res.writeHead(200, cache.headers);
|
|
return res.end(cache.body);
|
|
}
|
|
|
|
// send image files directly and let express handle content-length, etag, etc
|
|
if (mime.match(/^image/)) {
|
|
return res.sendFile(filePath, (err) => {
|
|
if (err && err.status === 404) {
|
|
// ensure we're triggering basic asset 404 and not a templated 404
|
|
return next(new errors.NotFoundError({
|
|
message: tpl(messages.imageNotFound),
|
|
code: 'STATIC_FILE_NOT_FOUND',
|
|
property: err.path
|
|
}));
|
|
}
|
|
|
|
if (err) {
|
|
return next(err);
|
|
}
|
|
});
|
|
}
|
|
|
|
// modify text files before caching+serving to ensure URL placeholders are transformed
|
|
fs.readFile(filePath, (err, buf) => {
|
|
if (err) {
|
|
// Downgrade to a simple 404 if the file didn't exist
|
|
if (err.code === 'ENOENT') {
|
|
err = new errors.NotFoundError({
|
|
message: tpl(messages.fileNotFound),
|
|
code: 'PUBLIC_FILE_NOT_FOUND',
|
|
property: err.path
|
|
});
|
|
}
|
|
return next(err);
|
|
}
|
|
|
|
let str = buf.toString();
|
|
|
|
if (mime === 'text/xsl' || mime === 'text/plain' || mime === 'application/javascript') {
|
|
str = str.replace(blogRegex, urlUtils.urlFor('home', true).replace(/\/$/, ''));
|
|
}
|
|
|
|
cache = {
|
|
headers: {
|
|
'Content-Type': mime,
|
|
'Content-Length': Buffer.from(str).length,
|
|
ETag: `"${crypto.createHash('md5').update(str, 'utf8').digest('hex')}"`,
|
|
'Cache-Control': `public, max-age=${maxAge}`
|
|
},
|
|
body: str,
|
|
key: req.query && req.query.v ? req.query.v : null
|
|
};
|
|
|
|
res.writeHead(200, cache.headers);
|
|
res.end(cache.body);
|
|
});
|
|
};
|
|
}
|
|
|
|
// ### servePublicFile Middleware
|
|
// Handles requests to robots.txt and favicon.ico (and caches them)
|
|
function servePublicFile(location, file, type, maxAge) {
|
|
const publicFileMiddleware = createPublicFileMiddleware(location, file, type, maxAge);
|
|
|
|
return function servePublicFileMiddleware(req, res, next) {
|
|
if (req.path === '/' + file) {
|
|
return publicFileMiddleware(req, res, next);
|
|
} else {
|
|
return next();
|
|
}
|
|
};
|
|
}
|
|
|
|
module.exports = servePublicFile;
|
|
module.exports.servePublicFile = servePublicFile;
|
|
module.exports.createPublicFileMiddleware = createPublicFileMiddleware;
|