2018-09-20 16:03:33 +03:00
|
|
|
const crypto = require('crypto');
|
|
|
|
const fs = require('fs-extra');
|
|
|
|
const path = require('path');
|
2020-04-09 21:40:00 +03:00
|
|
|
const errors = require('@tryghost/errors');
|
2020-05-27 20:47:53 +03:00
|
|
|
const config = require('../../../../shared/config');
|
2020-05-28 13:57:02 +03:00
|
|
|
const urlUtils = require('../../../../shared/url-utils');
|
2021-10-05 12:00:28 +03:00
|
|
|
const tpl = require('@tryghost/tpl');
|
|
|
|
|
|
|
|
const messages = {
|
|
|
|
imageNotFound: 'Image not found'
|
|
|
|
};
|
2015-07-15 19:01:23 +03:00
|
|
|
|
2019-04-23 17:24:33 +03:00
|
|
|
function createPublicFileMiddleware(file, type, maxAge) {
|
2018-09-10 15:07:57 +03:00
|
|
|
let content;
|
2018-09-20 16:03:33 +03:00
|
|
|
const publicFilePath = config.get('paths').publicFilePath;
|
|
|
|
const filePath = file.match(/^public/) ? path.join(publicFilePath, file.replace(/^public/, '')) : path.join(publicFilePath, file);
|
|
|
|
const blogRegex = /(\{\{blog-url\}\})/g;
|
2015-12-15 13:41:53 +03:00
|
|
|
|
2020-10-20 02:02:56 +03:00
|
|
|
return function servePublicFileMiddleware(req, res, next) {
|
2019-04-23 17:24:33 +03:00
|
|
|
if (content) {
|
|
|
|
res.writeHead(200, content.headers);
|
|
|
|
return res.end(content.body);
|
|
|
|
}
|
2020-02-10 12:51:19 +03:00
|
|
|
|
|
|
|
// send image files directly and let express handle content-length, etag, etc
|
|
|
|
if (type.match(/^image/)) {
|
|
|
|
return res.sendFile(filePath, (err) => {
|
|
|
|
if (err && err.status === 404) {
|
|
|
|
// ensure we're triggering basic asset 404 and not a templated 404
|
2020-04-09 21:40:00 +03:00
|
|
|
return next(new errors.NotFoundError({
|
2021-10-05 12:00:28 +03:00
|
|
|
message: tpl(messages.imageNotFound),
|
2020-02-10 12:51:19 +03:00
|
|
|
code: 'STATIC_FILE_NOT_FOUND',
|
|
|
|
property: err.path
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2020-02-17 02:24:01 +03:00
|
|
|
if (err) {
|
|
|
|
return next(err);
|
|
|
|
}
|
2020-02-10 12:51:19 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// modify text files before caching+serving to ensure URL placeholders are transformed
|
2019-04-23 17:24:33 +03:00
|
|
|
fs.readFile(filePath, (err, buf) => {
|
|
|
|
if (err) {
|
|
|
|
return next(err);
|
2015-07-15 19:01:23 +03:00
|
|
|
}
|
2019-04-23 17:24:33 +03:00
|
|
|
|
|
|
|
let str = buf.toString();
|
|
|
|
|
|
|
|
if (type === 'text/xsl' || type === 'text/plain' || type === 'application/javascript') {
|
2019-06-18 16:13:55 +03:00
|
|
|
str = str.replace(blogRegex, urlUtils.urlFor('home', true).replace(/\/$/, ''));
|
2019-04-23 17:24:33 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
content = {
|
|
|
|
headers: {
|
|
|
|
'Content-Type': type,
|
|
|
|
'Content-Length': Buffer.from(str).length,
|
|
|
|
ETag: `"${crypto.createHash('md5').update(str, 'utf8').digest('hex')}"`,
|
|
|
|
'Cache-Control': `public, max-age=${maxAge}`
|
|
|
|
},
|
|
|
|
body: str
|
|
|
|
};
|
|
|
|
res.writeHead(200, content.headers);
|
|
|
|
res.end(content.body);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// ### servePublicFile Middleware
|
|
|
|
// Handles requests to robots.txt and favicon.ico (and caches them)
|
|
|
|
function servePublicFile(file, type, maxAge) {
|
|
|
|
const publicFileMiddleware = createPublicFileMiddleware(file, type, maxAge);
|
|
|
|
|
2020-10-20 02:02:56 +03:00
|
|
|
return function servePublicFileMiddleware(req, res, next) {
|
2019-04-23 17:24:33 +03:00
|
|
|
if (req.path === '/' + file) {
|
|
|
|
return publicFileMiddleware(req, res, next);
|
2015-07-15 19:01:23 +03:00
|
|
|
} else {
|
2017-11-01 16:44:54 +03:00
|
|
|
return next();
|
2015-07-15 19:01:23 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-04-07 15:21:41 +03:00
|
|
|
module.exports = servePublicFile;
|
2019-04-23 17:24:33 +03:00
|
|
|
module.exports.servePublicFile = servePublicFile;
|
|
|
|
module.exports.createPublicFileMiddleware = createPublicFileMiddleware;
|