From 5199a007b52aff17300f506acc1478713062f660 Mon Sep 17 00:00:00 2001 From: Hannah Wolfe Date: Sun, 19 Oct 2014 21:10:13 +0200 Subject: [PATCH] frontend functional tests -> frontend route tests no issue - Some of the tests were duplicated and all of these tests are faster and easier to do with super test as we aren't testing interactions - Introduced a new test util to toggle permalinks which allows us to test the different structures - Using cheerio in route tests to test the HTML / XML output is well formed --- Gruntfile.js | 4 +- core/test/functional/frontend/error_test.js | 24 - core/test/functional/frontend/feed_test.js | 66 -- core/test/functional/frontend/home_test.js | 77 -- core/test/functional/frontend/post_test.js | 34 - core/test/functional/frontend/route_test.js | 10 - core/test/functional/routes/frontend_test.js | 973 +++++++++++-------- core/test/utils/index.js | 49 +- 8 files changed, 615 insertions(+), 622 deletions(-) delete mode 100644 core/test/functional/frontend/error_test.js delete mode 100644 core/test/functional/frontend/feed_test.js delete mode 100644 core/test/functional/frontend/home_test.js delete mode 100644 core/test/functional/frontend/post_test.js delete mode 100644 core/test/functional/frontend/route_test.js diff --git a/Gruntfile.js b/Gruntfile.js index 561405ad4e..6299d99b65 100644 --- a/Gruntfile.js +++ b/Gruntfile.js @@ -633,12 +633,12 @@ var _ = require('lodash'), // Custom test runner for our Casper.js functional tests // This really ought to be refactored into a separate grunt task module grunt.registerTask('spawnCasperJS', function (target) { - target = _.contains(['client', 'frontend', 'setup'], target) ? target + '/' : undefined; + target = _.contains(['client', 'setup'], target) ? target + '/' : undefined; var done = this.async(), options = ['host', 'noPort', 'port', 'email', 'password'], args = ['test'] - .concat(grunt.option('target') || target || ['client/', 'frontend/']) + .concat(grunt.option('target') || target || ['client/']) .concat(['--includes=base.js', '--log-level=debug', '--port=2369']); // Forward parameters from grunt to casperjs diff --git a/core/test/functional/frontend/error_test.js b/core/test/functional/frontend/error_test.js deleted file mode 100644 index 34e632b1e1..0000000000 --- a/core/test/functional/frontend/error_test.js +++ /dev/null @@ -1,24 +0,0 @@ -/** - * Tests if RSS exists and is working - */ -/*globals CasperTest, casper, url */ -CasperTest.begin('Check post not found (404)', 2, function suite(test) { - casper.thenOpen(url + 'asdf/', function (response) { - test.assertEqual(response.status, 404, 'Response status should be 404.'); - test.assertSelectorHasText('.error-code', '404'); - }); -}, true); - -CasperTest.begin('Check frontend route not found (404)', 2, function suite(test) { - casper.thenOpen(url + 'asdf/asdf/', function (response) { - test.assertEqual(response.status, 404, 'Response status should be 404.'); - test.assertSelectorHasText('.error-code', '404'); - }); -}, true); - -CasperTest.begin('Check frontend tag route not found (404)', 2, function suite(test) { - casper.thenOpen(url + 'tag/asdf/', function (response) { - test.assertEqual(response.status, 404, 'Response status should be 404.'); - test.assertSelectorHasText('.error-code', '404'); - }); -}, true); diff --git a/core/test/functional/frontend/feed_test.js b/core/test/functional/frontend/feed_test.js deleted file mode 100644 index d3ad3f8f32..0000000000 --- a/core/test/functional/frontend/feed_test.js +++ /dev/null @@ -1,66 +0,0 @@ -/** - * Tests if RSS exists and is working - */ -/*globals url, CasperTest, casper */ -CasperTest.begin('Ensure that RSS is available', 11, function suite(test) { - CasperTest.Routines.togglePermalinks.run('off'); - casper.thenOpen(url + 'rss/', function (response) { - var content = this.getHTML(), - siteTitle = '<![CDATA[Test Blog]]>', - siteDescription = '', - siteUrl = 'http://127.0.0.1:2369/', - postTitle = '', - postStart = 'You\'re live!', - postEnd = 'you think :)

]]>
', - postLink = 'http://127.0.0.1:2369/welcome-to-ghost/', - postCreator = ''; - - test.assertEqual(response.status, 200, 'Response status should be 200.'); - test.assert(content.indexOf('= 0, 'Feed should contain = 0, 'Feed should contain blog title.'); - test.assert(content.indexOf(siteDescription) >= 0, 'Feed should contain blog description.'); - test.assert(content.indexOf(siteUrl) >= 0, 'Feed should contain link to blog.'); - test.assert(content.indexOf(postTitle) >= 0, 'Feed should contain welcome post title.'); - test.assert(content.indexOf(postStart) >= 0, 'Feed should contain start of welcome post content.'); - test.assert(content.indexOf(postEnd) >= 0, 'Feed should contain end of welcome post content.'); - test.assert(content.indexOf(postLink) >= 0, 'Feed should have link to the welcome post.'); - test.assert(content.indexOf(postCreator) >= 0, 'Welcome post should have Test User as the creator.'); - test.assert(content.indexOf('') >= 0, 'Feed should contain '); - }); -}, false); - -CasperTest.begin('Ensure that author element is not included. Only dc:creator', 3, function suite(test) { - CasperTest.Routines.togglePermalinks.run('off'); - casper.thenOpen(url + 'rss/', function (response) { - var content = this.getHTML(), - author = '', - postCreator = ''; - - test.assertEqual(response.status, 200, 'Response status should be 200.'); - test.assert(content.indexOf(author) < 0, 'Author element should not be included'); - test.assert(content.indexOf(postCreator) >= 0, 'Welcome post should have Test User as the creator.'); - }); -}, false); - -CasperTest.begin('Ensures dated permalinks works with RSS', 2, function suite(test) { - CasperTest.Routines.togglePermalinks.run('on'); - casper.thenOpen(url + 'rss/', function (response) { - var content = this.getHTML(), - today = new Date(), - dd = ('0' + today.getDate()).slice(-2), - mm = ('0' + (today.getMonth() + 1)).slice(-2), - yyyy = today.getFullYear(), - postLink = '/' + yyyy + '/' + mm + '/' + dd + '/welcome-to-ghost/'; - - test.assertEqual(response.status, 200, 'Response status should be 200.'); - test.assert(content.indexOf(postLink) >= 0, 'Feed should have dated permalink.'); - }); - CasperTest.Routines.togglePermalinks.run('off'); -}, false); - -CasperTest.begin('Ensure that character set is UTF-8 for RSS feed', 1, function suite(test) { - CasperTest.Routines.togglePermalinks.run('off'); - casper.thenOpen(url + 'rss/', function (response) { - test.assertEqual(response.headers.get('Content-Type'), 'text/xml; charset=utf-8', 'Content type should include UTF-8 character set encoding.'); - }); -}, false); diff --git a/core/test/functional/frontend/home_test.js b/core/test/functional/frontend/home_test.js deleted file mode 100644 index 3cfebd3c75..0000000000 --- a/core/test/functional/frontend/home_test.js +++ /dev/null @@ -1,77 +0,0 @@ -/** - * Tests the homepage - */ - -/*globals CasperTest, casper, url */ -CasperTest.begin('Home page loads', 3, function suite(test) { - casper.start(url, function then() { - test.assertTitle('Test Blog', 'The homepage should have a title and it should be "Test Blog"'); - test.assertExists('.content .post', 'There is at least one post on this page'); - test.assertSelectorHasText('.poweredby', 'Proudly published with Ghost'); - }); -}, true); - -CasperTest.begin('Test helpers on homepage', 3, function suite(test) { - casper.start(url, function then() { - // body class - test.assertExists('body.home-template', 'body_class outputs correct home-template class'); - // post class - test.assertExists('article.post', 'post_class outputs correct post class'); - test.assertExists('article.tag-getting-started', 'post_class outputs correct tag class'); - }); -}, true); - -CasperTest.begin('Test navigating to Post', 4, function suite(test) { - casper.thenOpen(url, function then() { - var lastPost = '.content article:last-of-type', - lastPostLink = lastPost + ' .post-title a'; - - test.assertExists(lastPost, 'there is a last child on the page'); - test.assertSelectorHasText(lastPostLink, 'Welcome to Ghost', 'Is correct post'); - - casper.then(function testLink() { - var link = this.evaluate(function (lastPostLink) { - return document.querySelector(lastPostLink).getAttribute('href'); - }, lastPostLink); - - test.assert(link === '/welcome-to-ghost/', 'Has correct link'); - }); - - casper.thenClick(lastPostLink); - - casper.waitForResource(/welcome-to-ghost/).then(function (resource) { - test.assert(resource.status === 200, 'resource got 200'); - }); - }); -}, true); - -CasperTest.begin('Test navigating to Post with date permalink', 4, function suite(test) { - CasperTest.Routines.togglePermalinks.run('on'); - casper.thenOpen(url, function then() { - var lastPost = '.content article:last-of-type', - lastPostLink = lastPost + ' .post-title a', - today = new Date(), - dd = ('0' + today.getDate()).slice(-2), - mm = ('0' + (today.getMonth() + 1)).slice(-2), - yyyy = today.getFullYear(), - postLink = '/' + yyyy + '/' + mm + '/' + dd + '/welcome-to-ghost/'; - - test.assertExists(lastPost, 'there is a last child on the page'); - test.assertSelectorHasText(lastPostLink, 'Welcome to Ghost', 'Is correct post'); - - casper.then(function testLink() { - var link = this.evaluate(function (lastPostLink) { - return document.querySelector(lastPostLink).getAttribute('href'); - }, lastPostLink); - - test.assert(link === postLink, 'Has correct link'); - }); - - casper.thenClick(lastPostLink); - - casper.waitForResource(postLink).then(function (resource) { - test.assert(resource.status === 200, 'resource got 200'); - }); - }); - CasperTest.Routines.togglePermalinks.run('off'); -}, false); diff --git a/core/test/functional/frontend/post_test.js b/core/test/functional/frontend/post_test.js deleted file mode 100644 index 232c71efb1..0000000000 --- a/core/test/functional/frontend/post_test.js +++ /dev/null @@ -1,34 +0,0 @@ -/** - * Tests the default post page - */ - -/*globals CasperTest, casper, url */ - -// Tests when permalinks is set to date -CasperTest.begin('Post page does not load as slug', 2, function suite(test) { - CasperTest.Routines.togglePermalinks.run('on'); - casper.thenOpen(url + 'welcome-to-ghost', function then() { - test.assertTitle('404 — Page Not Found', 'The post should return 404 page'); - test.assertElementCount('.content .post', 0, 'There is no post on this page'); - }); - CasperTest.Routines.togglePermalinks.run('off'); -}, false); - -CasperTest.begin('Post page loads', 3, function suite(test) { - casper.thenOpen(url + 'welcome-to-ghost', function then() { - test.assertTitle('Welcome to Ghost', 'The post should have a title and it should be "Welcome to Ghost"'); - test.assertElementCount('.content .post', 1, 'There is exactly one post on this page'); - test.assertSelectorHasText('.poweredby', 'Proudly published with Ghost'); - }); -}, true); - -CasperTest.begin('Test helpers on welcome post', 4, function suite(test) { - casper.start(url + 'welcome-to-ghost', function then() { - // body class - test.assertExists('body.post-template', 'body_class outputs correct post-template class'); - test.assertExists('body.tag-getting-started', 'body_class outputs correct tag class'); - // post class - test.assertExists('article.post', 'post_class outputs correct post class'); - test.assertExists('article.tag-getting-started', 'post_class outputs correct tag class'); - }); -}, true); diff --git a/core/test/functional/frontend/route_test.js b/core/test/functional/frontend/route_test.js deleted file mode 100644 index d24bb5dc47..0000000000 --- a/core/test/functional/frontend/route_test.js +++ /dev/null @@ -1,10 +0,0 @@ -/** - * Tests archive page routing - */ - -/*globals CasperTest, casper, url */ -CasperTest.begin('Redirects page 1 request', 1, function suite(test) { - casper.thenOpen(url + 'page/1/', function then() { - test.assertEqual(casper.getCurrentUrl().indexOf('page/'), -1, 'Should be redirected to "/".'); - }); -}, true); diff --git a/core/test/functional/routes/frontend_test.js b/core/test/functional/routes/frontend_test.js index 2a1122d382..79a5e9fafb 100644 --- a/core/test/functional/routes/frontend_test.js +++ b/core/test/functional/routes/frontend_test.js @@ -8,6 +8,7 @@ var request = require('supertest'), should = require('should'), moment = require('moment'), + cheerio = require('cheerio'), testUtils = require('../../utils'), ghost = require('../../../../core'); @@ -40,189 +41,278 @@ describe('Frontend Routing', function () { }); }); - after(function (done) { - testUtils.clearData().then(function () { - done(); - }).catch(done); - }); + after(testUtils.teardown); - describe('Home', function () { - it('should respond with html', function (done) { - request.get('/') - .expect('Content-Type', /html/) - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) - .end(doEnd(done)); + describe('Test with Initial Fixtures', function () { + after(testUtils.teardown); + + describe('Error', function () { + it('should 404 for unknown post', function (done) { + request.get('/spectacular/') + .expect('Cache-Control', testUtils.cacheRules['private']) + .expect(404) + .expect(/Page Not Found/) + .end(doEnd(done)); + }); + + it('should 404 for unknown frontend route', function (done) { + request.get('/spectacular/marvellous/') + .expect('Cache-Control', testUtils.cacheRules['private']) + .expect(404) + .expect(/Page Not Found/) + .end(doEnd(done)); + }); + + it('should 404 for unknown tag', function (done) { + request.get('/tag/spectacular/') + .expect('Cache-Control', testUtils.cacheRules['private']) + .expect(404) + .expect(/Page Not Found/) + .end(doEnd(done)); + }); + + it('should 404 for unknown author', function (done) { + request.get('/author/spectacular/') + .expect('Cache-Control', testUtils.cacheRules['private']) + .expect(404) + .expect(/Page Not Found/) + .end(doEnd(done)); + }); }); - it('should not have as second page', function (done) { - request.get('/page/2/') - .expect('Location', '/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); + describe('Home', function () { + it('should respond with html', function (done) { + request.get('/') + .expect('Content-Type', /html/) + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(function (err, res) { + if (err) { + return done(err); + } + + var $ = cheerio.load(res.text); + + should.not.exist(res.headers['x-cache-invalidate']); + should.not.exist(res.headers['X-CSRF-Token']); + should.not.exist(res.headers['set-cookie']); + should.exist(res.headers.date); + + $('title').text().should.equal('Ghost'); + $('.content .post').length.should.equal(1); + $('.poweredby').text().should.equal('Proudly published with Ghost'); + $('body.home-template').length.should.equal(1); + $('article.post').length.should.equal(1); + $('article.tag-getting-started').length.should.equal(1); + + done(); + }); + }); + + it('should not have as second page', function (done) { + request.get('/page/2/') + .expect('Location', '/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + }); + + describe('Single post', function () { + it('should redirect without slash', function (done) { + request.get('/welcome-to-ghost') + .expect('Location', '/welcome-to-ghost/') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(301) + .end(doEnd(done)); + }); + + it('should redirect uppercase', function (done) { + request.get('/Welcome-To-Ghost/') + .expect('Location', '/welcome-to-ghost/') + .expect(301) + .end(doEnd(done)); + }); + + it('should respond with html for valid url', function (done) { + request.get('/welcome-to-ghost/') + .expect('Content-Type', /html/) + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(function (err, res) { + if (err) { + return done(err); + } + + var $ = cheerio.load(res.text); + + should.not.exist(res.headers['x-cache-invalidate']); + should.not.exist(res.headers['X-CSRF-Token']); + should.not.exist(res.headers['set-cookie']); + should.exist(res.headers.date); + + $('title').text().should.equal('Welcome to Ghost'); + $('.content .post').length.should.equal(1); + $('.poweredby').text().should.equal('Proudly published with Ghost'); + $('body.post-template').length.should.equal(1); + $('body.tag-getting-started').length.should.equal(1); + $('article.post').length.should.equal(1); + $('article.tag-getting-started').length.should.equal(1); + + done(); + }); + }); + + it('should not work with date permalinks', function (done) { + // get today's date + var date = moment().format('YYYY/MM/DD'); + + request.get('/' + date + '/welcome-to-ghost/') + .expect('Cache-Control', testUtils.cacheRules['private']) + .expect(404) + .expect(/Page Not Found/) + .end(doEnd(done)); + }); + }); + + describe('Post edit', function () { + it('should redirect without slash', function (done) { + request.get('/welcome-to-ghost/edit') + .expect('Location', '/welcome-to-ghost/edit/') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(301) + .end(doEnd(done)); + }); + + it('should redirect to editor', function (done) { + request.get('/welcome-to-ghost/edit/') + .expect('Location', '/ghost/editor/1/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + + it('should 404 for non-edit parameter', function (done) { + request.get('/welcome-to-ghost/notedit/') + .expect('Cache-Control', testUtils.cacheRules['private']) + .expect(404) + .expect(/Page Not Found/) + .end(doEnd(done)); + }); + }); + + describe('Static assets', function () { + it('should retrieve shared assets', function (done) { + request.get('/shared/img/user-image.png') + .expect('Cache-Control', testUtils.cacheRules.hour) + .expect(200) + .end(doEnd(done)); + }); + + it('should retrieve theme assets', function (done) { + request.get('/assets/css/screen.css') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(200) + .end(doEnd(done)); + }); + + it('should retrieve built assets', function (done) { + request.get('/ghost/scripts/vendor-dev.js') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(200) + .end(doEnd(done)); + }); + + it('should retrieve default robots.txt', function (done) { + request.get('/robots.txt') + .expect('Cache-Control', testUtils.cacheRules.hour) + .expect('ETag', /[0-9a-f]{32}/i) + .expect(200) + .end(doEnd(done)); + }); + + it('should retrieve default favicon.ico', function (done) { + request.get('/favicon.ico') + .expect('Cache-Control', testUtils.cacheRules.day) + .expect('ETag', /[0-9a-f]{32}/i) + .expect(200) + .end(doEnd(done)); + }); + + // at the moment there is no image fixture to test + // it('should retrieve image assets', function (done) { + // request.get('/content/images/some.jpg') + // .expect('Cache-Control', testUtils.cacheRules.year) + // .end(doEnd(done)); + // }); }); }); - describe('Welcome post', function () { - it('should redirect without slash', function (done) { - request.get('/welcome-to-ghost') - .expect('Location', '/welcome-to-ghost/') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(301) - .end(doEnd(done)); - }); - - it('should redirect uppercase', function (done) { - request.get('/Welcome-To-Ghost/') - .expect('Location', '/welcome-to-ghost/') - .expect(301) - .end(doEnd(done)); - }); - - it('should respond with html for valid url', function (done) { - request.get('/welcome-to-ghost/') - .expect('Content-Type', /html/) - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) - .end(doEnd(done)); - }); - - it('should not work with date permalinks', function (done) { - // get today's date - var date = moment().format('YYYY/MM/DD'); - - request.get('/' + date + '/welcome-to-ghost/') - // .expect('Cache-Control', testUtils.cacheRules['private']) - .expect(404) - .expect(/Page Not Found/) - .end(doEnd(done)); - }); - - it('should 404 for unknown post', function (done) { - request.get('/spectacular/') - .expect('Cache-Control', testUtils.cacheRules['private']) - .expect(404) - .expect(/Page Not Found/) - .end(doEnd(done)); - }); - }); - - describe('Post edit', function () { - it('should redirect without slash', function (done) { - request.get('/welcome-to-ghost/edit') - .expect('Location', '/welcome-to-ghost/edit/') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(301) - .end(doEnd(done)); - }); - - it('should redirect to editor', function (done) { - request.get('/welcome-to-ghost/edit/') - .expect('Location', '/ghost/editor/1/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); - - it('should 404 for non-edit parameter', function (done) { - request.get('/welcome-to-ghost/notedit/') - .expect('Cache-Control', testUtils.cacheRules['private']) - .expect(404) - .expect(/Page Not Found/) - .end(doEnd(done)); - }); - }); - - // we'll use X-Forwarded-Proto: https to simulate an 'https://' request behind a proxy - describe('HTTPS', function () { - var forkedGhost, request; + describe('Static page', function () { before(function (done) { - var configTestHttps = testUtils.fork.config(); - configTestHttps.forceAdminSSL = {redirect: false}; - configTestHttps.urlSSL = 'https://localhost/'; - - testUtils.fork.ghost(configTestHttps, 'testhttps') - .then(function (child) { - forkedGhost = child; - request = require('supertest'); - request = request(configTestHttps.url.replace(/\/$/, '')); - }).then(done).catch(done); + testUtils.initData().then(function () { + return testUtils.fixtures.insertPosts(); + }).then(function () { + done(); + }); }); - after(function (done) { - if (forkedGhost) { - forkedGhost.kill(done); - } - }); + after(testUtils.teardown); - it('should set links to url over non-HTTPS', function (done) { - request.get('/') - .expect(200) - .expect(//) - .expect(/Ghost<\/a\>/) - .end(doEnd(done)); - }); - - it('should set links to urlSSL over HTTPS', function (done) { - request.get('/') - .set('X-Forwarded-Proto', 'https') - .expect(200) - .expect(//) - .expect(/Ghost<\/a\>/) - .end(doEnd(done)); - }); - }); - - describe('RSS', function () { it('should redirect without slash', function (done) { - request.get('/rss') - .expect('Location', '/rss/') + request.get('/static-page-test') + .expect('Location', '/static-page-test/') .expect('Cache-Control', testUtils.cacheRules.year) .expect(301) .end(doEnd(done)); }); it('should respond with xml', function (done) { - request.get('/rss/') - .expect('Content-Type', /xml/) + request.get('/static-page-test/') + .expect('Content-Type', /html/) .expect('Cache-Control', testUtils.cacheRules['public']) .expect(200) .end(doEnd(done)); }); + }); - it('should not have as second page', function (done) { - request.get('/rss/2/') - // TODO this should probably redirect straight to /rss/ with 301? - .expect('Location', '/rss/1/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); + describe('Post with Ghost in the url', function () { + before(function (done) { + testUtils.initData().then(function () { + return testUtils.fixtures.insertPosts(); + }).then(function () { + done(); + }); }); - it('should get redirected to /rss/ from /feed/', function (done) { - request.get('/feed/') - .expect('Location', '/rss/') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(301) + after(testUtils.teardown); + + // All of Ghost's admin depends on the /ghost/ in the url to work properly + // Badly formed regexs can cause breakage if a post slug starts with the 5 letters ghost + it('should retrieve a blog post with ghost at the start of the url', function (done) { + request.get('/ghostly-kitchen-sink/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) .end(doEnd(done)); }); }); - // ### The rest of the tests require more data - describe('Archive pages', function () { // Add enough posts to trigger pages for both the archive (5 pp) and rss (15 pp) // insertPosts adds 5 published posts, 1 draft post, 1 published static page and one draft page // we then insert with max 11 which ensures we have 16 published posts before(function (done) { - testUtils.fixtures.insertPosts().then(function () { + testUtils.initData().then(function () { + return testUtils.fixtures.insertPosts(); + }).then(function () { return testUtils.fixtures.insertMorePosts(9); }).then(function () { done(); }).catch(done); }); + after(testUtils.teardown); + it('should redirect without slash', function (done) { request.get('/page/2') .expect('Location', '/page/2/') @@ -265,271 +355,133 @@ describe('Frontend Routing', function () { }); }); - describe('RSS pages', function () { + describe('RSS', function () { before(function (done) { - testUtils.fixtures.insertMorePosts(2).then(function () { + testUtils.initData().then(function () { + return testUtils.fixtures.overrideOwnerUser(); + }).then(function () { done(); - }).catch(done); + }); }); + + after(testUtils.teardown); + it('should redirect without slash', function (done) { - request.get('/rss/2') - .expect('Location', '/rss/2/') + request.get('/rss') + .expect('Location', '/rss/') .expect('Cache-Control', testUtils.cacheRules.year) .expect(301) .end(doEnd(done)); }); it('should respond with xml', function (done) { + request.get('/rss/') + .expect('Content-Type', 'text/xml; charset=utf-8') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(function (err, res) { + if (err) { + return done(err); + } + + should.not.exist(res.headers['x-cache-invalidate']); + should.not.exist(res.headers['X-CSRF-Token']); + should.not.exist(res.headers['set-cookie']); + should.exist(res.headers.date); + + var content = res.text, + siteTitle = '<![CDATA[Ghost]]>', + siteDescription = '', + siteUrl = 'http://127.0.0.1:2369/', + postTitle = '', + postStart = 'You\'re live!', + postEnd = 'you think :)

]]>
', + postLink = 'http://127.0.0.1:2369/welcome-to-ghost/', + postCreator = '', + author = ''; + + content.indexOf('').should.be.above(0); + content.indexOf(author).should.be.below(0); + content.indexOf(postCreator).should.be.above(0); + + done(); + }); + }); + + it('should not have as second page', function (done) { request.get('/rss/2/') - .expect('Content-Type', /xml/) + // TODO this should probably redirect straight to /rss/ with 301? + .expect('Location', '/rss/1/') .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) + .expect(302) .end(doEnd(done)); }); - it('should redirect page 1', function (done) { - request.get('/rss/1/') + it('should get redirected to /rss/ from /feed/', function (done) { + request.get('/feed/') .expect('Location', '/rss/') - .expect('Cache-Control', testUtils.cacheRules['public']) - // TODO: This should probably be a 301? - .expect(302) - .end(doEnd(done)); - }); - - it('should redirect to last page if page too high', function (done) { - request.get('/rss/3/') - .expect('Location', '/rss/2/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); - - it('should redirect to first page if page too low', function (done) { - request.get('/rss/0/') - .expect('Location', '/rss/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); - }); - - describe('Tag based RSS pages', function () { - it('should redirect without slash', function (done) { - request.get('/tag/getting-started/rss') - .expect('Location', '/tag/getting-started/rss/') .expect('Cache-Control', testUtils.cacheRules.year) .expect(301) .end(doEnd(done)); }); - it('should respond with xml', function (done) { - request.get('/tag/getting-started/rss/') - .expect('Content-Type', /xml/) - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) - .end(doEnd(done)); - }); + describe('RSS pages', function () { + before(function (done) { + testUtils.fixtures.insertPosts().then(function () { + return testUtils.fixtures.insertMorePosts(11); + }).then(function () { + done(); + }).catch(done); + }); + it('should redirect without slash', function (done) { + request.get('/rss/2') + .expect('Location', '/rss/2/') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(301) + .end(doEnd(done)); + }); - it('should redirect page 1', function (done) { - request.get('/tag/getting-started/rss/1/') - .expect('Location', '/tag/getting-started/rss/') - .expect('Cache-Control', testUtils.cacheRules['public']) - // TODO: This should probably be a 301? - .expect(302) - .end(doEnd(done)); - }); + it('should respond with xml', function (done) { + request.get('/rss/2/') + .expect('Content-Type', /xml/) + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(doEnd(done)); + }); - it('should redirect to last page if page too high', function (done) { - request.get('/tag/getting-started/rss/2/') - .expect('Location', '/tag/getting-started/rss/1/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); + it('should redirect page 1', function (done) { + request.get('/rss/1/') + .expect('Location', '/rss/') + .expect('Cache-Control', testUtils.cacheRules['public']) + // TODO: This should probably be a 301? + .expect(302) + .end(doEnd(done)); + }); - it('should redirect to first page if page too low', function (done) { - request.get('/tag/getting-started/rss/0/') - .expect('Location', '/tag/getting-started/rss/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); - }); + it('should redirect to last page if page too high', function (done) { + request.get('/rss/3/') + .expect('Location', '/rss/2/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); - describe('Author based RSS pages', function () { - it('should redirect without slash', function (done) { - request.get('/author/ghost-owner/rss') - .expect('Location', '/author/ghost-owner/rss/') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(301) - .end(doEnd(done)); - }); - - it('should respond with xml', function (done) { - request.get('/author/ghost-owner/rss/') - .expect('Content-Type', /xml/) - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) - .end(doEnd(done)); - }); - - it('should redirect page 1', function (done) { - request.get('/author/ghost-owner/rss/1/') - .expect('Location', '/author/ghost-owner/rss/') - .expect('Cache-Control', testUtils.cacheRules['public']) - // TODO: This should probably be a 301? - .expect(302) - .end(doEnd(done)); - }); - - it('should redirect to last page if page too high', function (done) { - request.get('/author/ghost-owner/rss/3/') - .expect('Location', '/author/ghost-owner/rss/2/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); - - it('should redirect to first page if page too low', function (done) { - request.get('/author/ghost-owner/rss/0/') - .expect('Location', '/author/ghost-owner/rss/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); - }); - - describe('Static page', function () { - it('should redirect without slash', function (done) { - request.get('/static-page-test') - .expect('Location', '/static-page-test/') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(301) - .end(doEnd(done)); - }); - - it('should respond with xml', function (done) { - request.get('/static-page-test/') - .expect('Content-Type', /html/) - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) - .end(doEnd(done)); - }); - }); - - describe('Post with Ghost in the url', function () { - // All of Ghost's admin depends on the /ghost/ in the url to work properly - // Badly formed regexs can cause breakage if a post slug starts with the 5 letters ghost - it('should retrieve a blog post with ghost at the start of the url', function (done) { - request.get('/ghostly-kitchen-sink/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) - .end(doEnd(done)); - }); - }); - - describe('Static assets', function () { - it('should retrieve shared assets', function (done) { - request.get('/shared/img/user-image.png') - .expect('Cache-Control', testUtils.cacheRules.hour) - .expect(200) - .end(doEnd(done)); - }); - - it('should retrieve theme assets', function (done) { - request.get('/assets/css/screen.css') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(200) - .end(doEnd(done)); - }); - - it('should retrieve built assets', function (done) { - request.get('/ghost/scripts/vendor-dev.js') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(200) - .end(doEnd(done)); - }); - - it('should retrieve default robots.txt', function (done) { - request.get('/robots.txt') - .expect('Cache-Control', testUtils.cacheRules.hour) - .expect('ETag', /[0-9a-f]{32}/i) - .expect(200) - .end(doEnd(done)); - }); - - it('should retrieve default favicon.ico', function (done) { - request.get('/favicon.ico') - .expect('Cache-Control', testUtils.cacheRules.day) - .expect('ETag', /[0-9a-f]{32}/i) - .expect(200) - .end(doEnd(done)); - }); - - // at the moment there is no image fixture to test - // it('should retrieve image assets', function (done) { - // request.get('/content/images/some.jpg') - // .expect('Cache-Control', testUtils.cacheRules.year) - // .end(doEnd(done)); - // }); - }); - - describe('Tag pages', function () { - // Add enough posts to trigger tag pages - before(function (done) { - testUtils.clearData().then(function () { - // we initialise data, but not a user. No user should be required for navigating the frontend - return testUtils.initData(); - }).then(function () { - return testUtils.fixtures.insertPosts(); - }).then(function () { - return testUtils.fixtures.insertMorePosts(22); - }).then(function () { - return testUtils.fixtures.insertMorePostsTags(22); - }).then(function () { - done(); - }).catch(done); - }); - - it('should redirect without slash', function (done) { - request.get('/tag/injection/page/2') - .expect('Location', '/tag/injection/page/2/') - .expect('Cache-Control', testUtils.cacheRules.year) - .expect(301) - .end(doEnd(done)); - }); - - it('should respond with html', function (done) { - request.get('/tag/injection/page/2/') - .expect('Content-Type', /html/) - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(200) - .end(doEnd(done)); - }); - - it('should redirect page 1', function (done) { - request.get('/tag/injection/page/1/') - .expect('Location', '/tag/injection/') - .expect('Cache-Control', testUtils.cacheRules['public']) - // TODO: This should probably be a 301? - .expect(302) - .end(doEnd(done)); - }); - - it('should redirect to last page if page too high', function (done) { - request.get('/tag/injection/page/4/') - .expect('Location', '/tag/injection/page/3/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); - }); - - it('should redirect to first page if page too low', function (done) { - request.get('/tag/injection/page/0/') - .expect('Location', '/tag/injection/') - .expect('Cache-Control', testUtils.cacheRules['public']) - .expect(302) - .end(doEnd(done)); + it('should redirect to first page if page too low', function (done) { + request.get('/rss/0/') + .expect('Location', '/rss/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); }); }); @@ -542,12 +494,14 @@ describe('Frontend Routing', function () { }).then(function () { return testUtils.fixtures.insertPosts(); }).then(function () { - return testUtils.fixtures.insertMorePosts(10); + return testUtils.fixtures.insertMorePosts(9); }).then(function () { done(); }).catch(done); }); + after(testUtils.teardown); + it('should redirect without slash', function (done) { request.get('/author/ghost-owner/page/2') .expect('Location', '/author/ghost-owner/page/2/') @@ -588,26 +542,237 @@ describe('Frontend Routing', function () { .expect(302) .end(doEnd(done)); }); + + describe('Author based RSS pages', function () { + it('should redirect without slash', function (done) { + request.get('/author/ghost-owner/rss') + .expect('Location', '/author/ghost-owner/rss/') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(301) + .end(doEnd(done)); + }); + + it('should respond with xml', function (done) { + request.get('/author/ghost-owner/rss/') + .expect('Content-Type', /xml/) + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(doEnd(done)); + }); + + it('should redirect page 1', function (done) { + request.get('/author/ghost-owner/rss/1/') + .expect('Location', '/author/ghost-owner/rss/') + .expect('Cache-Control', testUtils.cacheRules['public']) + // TODO: This should probably be a 301? + .expect(302) + .end(doEnd(done)); + }); + + it('should redirect to last page if page too high', function (done) { + request.get('/author/ghost-owner/rss/2/') + .expect('Location', '/author/ghost-owner/rss/1/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + + it('should redirect to first page if page too low', function (done) { + request.get('/author/ghost-owner/rss/0/') + .expect('Location', '/author/ghost-owner/rss/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + }); }); - // ### The rest of the tests switch to date permalinks + describe('Tag pages', function () { + // Add enough posts to trigger tag pages + before(function (done) { + testUtils.initData().then(function () { + return testUtils.fixtures.insertPosts(); + }).then(function () { + return testUtils.fixtures.insertMorePosts(22); + }).then(function () { + return testUtils.fixtures.insertMorePostsTags(22); + }).then(function () { + done(); + }).catch(done); + }); -// describe('Date permalinks', function () { -// before(function (done) { -// // Only way to swap permalinks setting is to login and visit the URL because -// // poking the database doesn't work as settings are cached -// }); -// -// it('should load a post with date permalink', function (done) { -// -// // get today's date -// var date = moment().format('YYYY/MM/DD'); -// -// -// request.get('/' + date + '/welcome-to-ghost/') -// .expect(200) -// .expect('Content-Type', /html/) -// .end(doEnd(done)); -// }); -// }); + after(testUtils.teardown); + + it('should redirect without slash', function (done) { + request.get('/tag/injection/page/2') + .expect('Location', '/tag/injection/page/2/') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(301) + .end(doEnd(done)); + }); + + it('should respond with html', function (done) { + request.get('/tag/injection/page/2/') + .expect('Content-Type', /html/) + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(doEnd(done)); + }); + + it('should redirect page 1', function (done) { + request.get('/tag/injection/page/1/') + .expect('Location', '/tag/injection/') + .expect('Cache-Control', testUtils.cacheRules['public']) + // TODO: This should probably be a 301? + .expect(302) + .end(doEnd(done)); + }); + + it('should redirect to last page if page too high', function (done) { + request.get('/tag/injection/page/4/') + .expect('Location', '/tag/injection/page/3/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + + it('should redirect to first page if page too low', function (done) { + request.get('/tag/injection/page/0/') + .expect('Location', '/tag/injection/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + + describe('Tag based RSS pages', function () { + it('should redirect without slash', function (done) { + request.get('/tag/getting-started/rss') + .expect('Location', '/tag/getting-started/rss/') + .expect('Cache-Control', testUtils.cacheRules.year) + .expect(301) + .end(doEnd(done)); + }); + + it('should respond with xml', function (done) { + request.get('/tag/getting-started/rss/') + .expect('Content-Type', /xml/) + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(doEnd(done)); + }); + + it('should redirect page 1', function (done) { + request.get('/tag/getting-started/rss/1/') + .expect('Location', '/tag/getting-started/rss/') + .expect('Cache-Control', testUtils.cacheRules['public']) + // TODO: This should probably be a 301? + .expect(302) + .end(doEnd(done)); + }); + + it('should redirect to last page if page too high', function (done) { + request.get('/tag/getting-started/rss/2/') + .expect('Location', '/tag/getting-started/rss/1/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + + it('should redirect to first page if page too low', function (done) { + request.get('/tag/getting-started/rss/0/') + .expect('Location', '/tag/getting-started/rss/') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(302) + .end(doEnd(done)); + }); + }); + }); + + // we'll use X-Forwarded-Proto: https to simulate an 'https://' request behind a proxy + describe('HTTPS', function () { + var forkedGhost, request; + before(function (done) { + var configTestHttps = testUtils.fork.config(); + configTestHttps.forceAdminSSL = {redirect: false}; + configTestHttps.urlSSL = 'https://localhost/'; + + testUtils.fork.ghost(configTestHttps, 'testhttps') + .then(function (child) { + forkedGhost = child; + request = require('supertest'); + request = request(configTestHttps.url.replace(/\/$/, '')); + }).then(done).catch(done); + }); + + after(function (done) { + if (forkedGhost) { + forkedGhost.kill(done); + } + }); + + it('should set links to url over non-HTTPS', function (done) { + request.get('/') + .expect(200) + .expect(//) + .expect(/
Ghost<\/a\>/) + .end(doEnd(done)); + }); + + it('should set links to urlSSL over HTTPS', function (done) { + request.get('/') + .set('X-Forwarded-Proto', 'https') + .expect(200) + .expect(//) + .expect(/Ghost<\/a\>/) + .end(doEnd(done)); + }); + }); + + describe('Date permalinks', function () { + before(function (done) { + // Only way to swap permalinks setting is to login and visit the URL because + // poking the database doesn't work as settings are cached + testUtils.togglePermalinks(request, 'date').then(function () { + done(); + }); + }); + + it('should load a post with date permalink', function (done) { + // get today's date + var date = moment().format('YYYY/MM/DD'); + + request.get('/' + date + '/welcome-to-ghost/') + .expect(200) + .expect('Content-Type', /html/) + .end(doEnd(done)); + }); + + it('should serve RSS with date permalink', function (done) { + request.get('/rss/') + .expect('Content-Type', 'text/xml; charset=utf-8') + .expect('Cache-Control', testUtils.cacheRules['public']) + .expect(200) + .end(function (err, res) { + if (err) { + return done(err); + } + + should.not.exist(res.headers['x-cache-invalidate']); + should.not.exist(res.headers['X-CSRF-Token']); + should.not.exist(res.headers['set-cookie']); + should.exist(res.headers.date); + + var content = res.text, + today = new Date(), + dd = ('0' + today.getDate()).slice(-2), + mm = ('0' + (today.getMonth() + 1)).slice(-2), + yyyy = today.getFullYear(), + postLink = '/' + yyyy + '/' + mm + '/' + dd + '/welcome-to-ghost/'; + + content.indexOf(postLink).should.be.above(0); + + done(); + }); + }); + }); }); diff --git a/core/test/utils/index.js b/core/test/utils/index.js index 387e6f1b0d..94a8798081 100644 --- a/core/test/utils/index.js +++ b/core/test/utils/index.js @@ -21,6 +21,8 @@ var Promise = require('bluebird'), teardown, setup, doAuth, + login, + togglePermalinks, initFixtures, initData, @@ -462,7 +464,6 @@ setup = function setup() { doAuth = function doAuth() { var options = arguments, request = arguments[0], - user = DataGenerator.forModel.users[0], fixtureOps; // Remove request from this list @@ -474,16 +475,52 @@ doAuth = function doAuth() { fixtureOps = getFixtureOps(options); + return sequence(fixtureOps).then(function () { + return login(request); + }); +}; + +login = function login(request) { + var user = DataGenerator.forModel.users[0]; + return new Promise(function (resolve, reject) { - return sequence(fixtureOps).then(function () { - request.post('/ghost/api/v0.1/authentication/token/') - .send({grant_type: 'password', username: user.email, password: user.password, client_id: 'ghost-admin'}) + request.post('/ghost/api/v0.1/authentication/token/') + .send({grant_type: 'password', username: user.email, password: user.password, client_id: 'ghost-admin'}) + .end(function (err, res) { + if (err) { + return reject(err); + } + + resolve(res.body.access_token); + }); + }); +}; + +togglePermalinks = function togglePermalinks(request, toggle) { + var permalinkString = toggle === 'date' ? '/:year/:month/:day/:slug/' : '/:slug/'; + + return new Promise(function (resolve, reject) { + doAuth(request).then(function (token) { + request.put('/ghost/api/v0.1/settings/') + .set('Authorization', 'Bearer ' + token) + .send({settings: [ + { + uuid: '75e994ae-490e-45e6-9207-0eab409c1c04', + key: 'permalinks', + value: permalinkString, + type: 'blog', + created_at: '2014-10-16T17:39:16.005Z', + created_by: 1, + updated_at: '2014-10-20T19:44:18.077Z', + updated_by: 1 + } + ]}) .end(function (err, res) { if (err) { return reject(err); } - resolve(res.body.access_token); + resolve(res.body); }); }); }); @@ -499,6 +536,8 @@ module.exports = { teardown: teardown, setup: setup, doAuth: doAuth, + login: login, + togglePermalinks: togglePermalinks, initFixtures: initFixtures, initData: initData,