0
Fork 0
mirror of https://github.com/TryGhost/Ghost.git synced 2025-04-01 02:41:39 -05:00

🐛 Fixed private blogging leaking post information (#8999)

* 🐛  Fixed private blogging leaking post information

closes #8990

- a condition in the private blogging app redirected rss && sitemap to 404, which can possibly leak content
- remove this condition and ensure we always redirect to /private

* lint 😋
This commit is contained in:
Katharina Irrgang 2017-09-11 15:09:19 +02:00 committed by Hannah Wolfe
parent 7e211a307c
commit add9e541c2
2 changed files with 39 additions and 49 deletions

View file

@ -6,7 +6,6 @@ var _ = require('lodash'),
Promise = require('bluebird'),
config = require('../../../config'),
api = require('../../../api'),
errors = require('../../../errors'),
utils = require('../../../utils'),
i18n = require('../../../i18n'),
privateRoute = '/' + config.get('routeKeywords').private + '/',
@ -50,12 +49,7 @@ privateBlogging = {
return next();
}
// take care of rss and sitemap 404s
if (req.path.lastIndexOf('/rss/', 0) === 0 ||
req.path.lastIndexOf('/rss/') === req.url.length - 5 ||
(req.path.lastIndexOf('/sitemap', 0) === 0 && req.path.lastIndexOf('.xml') === req.path.length - 4)) {
return next(new errors.NotFoundError({message: i18n.t('errors.errors.pageNotFound')}));
} else if (req.url.lastIndexOf('/robots.txt', 0) === 0) {
if (req.url.lastIndexOf('/robots.txt', 0) === 0) {
fs.readFile(path.resolve(__dirname, '../', 'robots.txt'), function readFile(err, buf) {
if (err) {
return next(err);

View file

@ -4,7 +4,6 @@ var should = require('should'), // jshint ignore:line
crypto = require('crypto'),
Promise = require('bluebird'),
api = require('../../../api'),
errors = require('../../../errors'),
fs = require('fs'),
privateBlogging = require('../lib/middleware'),
@ -33,7 +32,7 @@ describe('Private Blogging', function () {
req = {};
res = {};
apiSettingsStub = sandbox.stub(api.settings, 'read');
next = sinon.spy();
next = sandbox.spy();
});
it('checkIsPrivate should call next if not private', function (done) {
@ -79,7 +78,7 @@ describe('Private Blogging', function () {
it('isPrivateSessionAuth should redirect if blog is not private', function () {
res = {
redirect: sinon.spy(),
redirect: sandbox.spy(),
isPrivateBlog: false
};
privateBlogging.isPrivateSessionAuth(req, res, next);
@ -89,7 +88,6 @@ describe('Private Blogging', function () {
describe('private', function () {
beforeEach(function () {
res.isPrivateBlog = true;
res = {
status: function () {
return this;
@ -98,8 +96,11 @@ describe('Private Blogging', function () {
},
set: function () {
},
redirect: sandbox.spy(),
isPrivateBlog: true
};
req.session = {};
});
it('filterPrivateRoutes should call next if admin', function () {
@ -114,70 +115,65 @@ describe('Private Blogging', function () {
next.called.should.be.true();
});
it('filterPrivateRoutes should throw 404 if url is sitemap', function () {
it('filterPrivateRoutes: sitemap redirects to /private', function () {
req.path = req.url = '/sitemap.xml';
var next = function next(err) {
(err instanceof errors.NotFoundError).should.eql(true);
};
privateBlogging.filterPrivateRoutes(req, res, next);
return privateBlogging.filterPrivateRoutes(req, res, next)
.then(function () {
res.redirect.calledOnce.should.be.true();
});
});
it('filterPrivateRoutes should throw 404 if url is sitemap with param', function () {
it('filterPrivateRoutes: sitemap with params redirects to /private', function () {
req.url = '/sitemap.xml?weird=param';
req.path = '/sitemap.xml';
var next = function next(err) {
(err instanceof errors.NotFoundError).should.eql(true);
};
privateBlogging.filterPrivateRoutes(req, res, next);
return privateBlogging.filterPrivateRoutes(req, res, next)
.then(function () {
res.redirect.calledOnce.should.be.true();
});
});
it('filterPrivateRoutes should throw 404 if url is rss', function () {
it('filterPrivateRoutes: rss redirects to /private', function () {
req.path = req.url = '/rss/';
var next = function next(err) {
(err instanceof errors.NotFoundError).should.eql(true);
};
privateBlogging.filterPrivateRoutes(req, res, next);
return privateBlogging.filterPrivateRoutes(req, res, next)
.then(function () {
res.redirect.calledOnce.should.be.true();
});
});
it('filterPrivateRoutes should throw 404 if url is author rss', function () {
it('filterPrivateRoutes: author rss redirects to /private', function () {
req.path = req.url = '/author/halfdan/rss/';
var next = function next(err) {
(err instanceof errors.NotFoundError).should.eql(true);
};
privateBlogging.filterPrivateRoutes(req, res, next);
return privateBlogging.filterPrivateRoutes(req, res, next)
.then(function () {
res.redirect.calledOnce.should.be.true();
});
});
it('filterPrivateRoutes should throw 404 if url is tag rss', function () {
it('filterPrivateRoutes: tag rss redirects to /private', function () {
req.path = req.url = '/tag/slimer/rss/';
var next = function next(err) {
(err instanceof errors.NotFoundError).should.eql(true);
};
privateBlogging.filterPrivateRoutes(req, res, next);
return privateBlogging.filterPrivateRoutes(req, res, next)
.then(function () {
res.redirect.calledOnce.should.be.true();
});
});
it('filterPrivateRoutes should throw 404 if url is rss plus something', function () {
it('filterPrivateRoutes: rss plus something redirects to /private', function () {
req.path = req.url = '/rss/sometag';
var next = function next(err) {
(err instanceof errors.NotFoundError).should.eql(true);
};
privateBlogging.filterPrivateRoutes(req, res, next);
return privateBlogging.filterPrivateRoutes(req, res, next)
.then(function () {
res.redirect.calledOnce.should.be.true();
});
});
it('filterPrivateRoutes should render custom robots.txt', function () {
req.url = req.path = '/robots.txt';
res.writeHead = sinon.spy();
res.end = sinon.spy();
res.writeHead = sandbox.spy();
res.end = sandbox.spy();
sandbox.stub(fs, 'readFile', function (file, cb) {
cb(null, 'User-agent: * Disallow: /');
});
@ -223,7 +219,7 @@ describe('Private Blogging', function () {
token: 'wrongpassword',
salt: Date.now().toString()
};
res.redirect = sinon.spy();
res.redirect = sandbox.spy();
privateBlogging.authenticatePrivateSession(req, res, next).then(function () {
res.redirect.called.should.be.true();