mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-06 22:40:14 -05:00
Cleaned up test utils with async/await
- I'm on a mission to make this code comprehensible so we can work it into something better with new boot - Who else loves async/await? :D - Dried up a block of duplicated code
This commit is contained in:
parent
2c96df42ac
commit
24bfb5567b
4 changed files with 164 additions and 147 deletions
66
test/new/magic_spec.js
Normal file
66
test/new/magic_spec.js
Normal file
|
@ -0,0 +1,66 @@
|
|||
// # Default Frontend Routing Test
|
||||
// These tests check the default out-of-the-box behaviour of Ghost is working as expected.
|
||||
|
||||
// Test Structure
|
||||
// As it stands, these tests depend on the database, and as such are integration tests.
|
||||
// Mocking out the models to not touch the DB would turn these into unit tests, and should probably be done in future,
|
||||
// But then again testing real code, rather than mock code, might be more useful...
|
||||
const should = require('should');
|
||||
const sinon = require('sinon');
|
||||
const supertest = require('supertest');
|
||||
const testUtils = require('./utils');
|
||||
const configUtils = require('../utils/configUtils');
|
||||
|
||||
describe('Default Frontend routing', function () {
|
||||
let request;
|
||||
|
||||
afterEach(function () {
|
||||
sinon.restore();
|
||||
});
|
||||
|
||||
before(async function () {
|
||||
await testUtils.startGhost();
|
||||
request = supertest.agent(configUtils.config.get('url'));
|
||||
});
|
||||
|
||||
describe('Main Routes', function () {
|
||||
it('/ should respond with valid HTML', async function () {
|
||||
await request.get('/')
|
||||
.expect((res) => {
|
||||
console.log(res.text);
|
||||
})
|
||||
.expect(200)
|
||||
.expect('Content-Type', /html/)
|
||||
.expect('Cache-Control', testUtils.cacheRules.public);
|
||||
|
||||
// const $ = cheerio.load(res.text);
|
||||
|
||||
// // NOTE: "Ghost" is the title from the settings.
|
||||
// $('title').text().should.equal('Ghost');
|
||||
|
||||
// $('body.home-template').length.should.equal(1);
|
||||
// $('article.post').length.should.equal(7);
|
||||
// $('article.tag-getting-started').length.should.equal(7);
|
||||
|
||||
// doEnd(res);
|
||||
});
|
||||
|
||||
it('/author/ghost/ should respond with valid HTML', async function () {
|
||||
await request.get('/author/ghost/')
|
||||
.expect('Content-Type', /html/);
|
||||
// .expect('Cache-Control', testUtils.cacheRules.public)
|
||||
// .expect(200);
|
||||
|
||||
// const $ = cheerio.load(res.text);
|
||||
|
||||
// // NOTE: "Ghost" is the title from the settings.
|
||||
// $('title').text().should.equal('Ghost - Ghost');
|
||||
|
||||
// $('body.author-template').length.should.equal(1);
|
||||
// $('article.post').length.should.equal(7);
|
||||
// $('article.tag-getting-started').length.should.equal(7);
|
||||
|
||||
// doEnd(res);
|
||||
});
|
||||
});
|
||||
});
|
14
test/new/utils.js
Normal file
14
test/new/utils.js
Normal file
|
@ -0,0 +1,14 @@
|
|||
const boot = require('../../core/boot');
|
||||
const urlServiceUtils = require('../utils/url-service-utils');
|
||||
const cacheRules = require('../utils/fixtures/cache-rules');
|
||||
|
||||
module.exports.startGhost = async () => {
|
||||
const startTime = Date.now();
|
||||
await boot();
|
||||
console.log(`Ghost booted in ${(Date.now() - startTime) / 1000}s`); // eslint-disable-line no-console
|
||||
|
||||
await urlServiceUtils.isFinished();
|
||||
console.log(`Ghost ready in ${(Date.now() - startTime) / 1000}s`); // eslint-disable-line no-console
|
||||
};
|
||||
|
||||
module.exports.cacheRules = cacheRules;
|
|
@ -15,38 +15,27 @@ const schemaTables = Object.keys(schema);
|
|||
// Other Test Utilities
|
||||
const urlServiceUtils = require('./url-service-utils');
|
||||
|
||||
module.exports.initData = () => {
|
||||
return knexMigrator.init()
|
||||
.then(function () {
|
||||
return urlServiceUtils.isFinished();
|
||||
});
|
||||
module.exports.initData = async () => {
|
||||
await knexMigrator.init();
|
||||
await urlServiceUtils.isFinished();
|
||||
};
|
||||
|
||||
module.exports.clearBruteData = () => {
|
||||
return db.knex('brute').truncate();
|
||||
};
|
||||
|
||||
module.exports.truncate = (tableName) => {
|
||||
module.exports.truncate = async (tableName) => {
|
||||
if (config.get('database:client') === 'sqlite3') {
|
||||
return db.knex(tableName).truncate();
|
||||
await db.knex(tableName).truncate();
|
||||
return;
|
||||
}
|
||||
|
||||
return db.knex.raw('SET FOREIGN_KEY_CHECKS=0;')
|
||||
.then(function () {
|
||||
return db.knex(tableName).truncate();
|
||||
})
|
||||
.then(function () {
|
||||
return db.knex.raw('SET FOREIGN_KEY_CHECKS=1;');
|
||||
});
|
||||
await db.knex.raw('SET FOREIGN_KEY_CHECKS=0;');
|
||||
await db.knex(tableName).truncate();
|
||||
await db.knex.raw('SET FOREIGN_KEY_CHECKS=1;');
|
||||
};
|
||||
|
||||
// we must always try to delete all tables
|
||||
module.exports.clearData = () => {
|
||||
module.exports.clearData = async () => {
|
||||
debug('Database reset');
|
||||
return knexMigrator.reset({force: true})
|
||||
.then(function () {
|
||||
urlServiceUtils.reset();
|
||||
});
|
||||
await knexMigrator.reset({force: true});
|
||||
urlServiceUtils.reset();
|
||||
};
|
||||
|
||||
/**
|
||||
|
|
|
@ -115,6 +115,34 @@ const createEmailedPost = async function createEmailedPost({postOptions, emailOp
|
|||
|
||||
let ghostServer;
|
||||
|
||||
const dirtyDataFunction = () => {
|
||||
/**
|
||||
* @TODO: this is dirty, but makes routing testing a lot easier for now, because the routing test
|
||||
* has no easy way to access existing resource id's, which are added from the Ghost fixtures.
|
||||
* I can do `testUtils.existingData.roles[0].id`.
|
||||
*/
|
||||
module.exports.existingData = {};
|
||||
return models.Role.findAll({columns: ['id']})
|
||||
.then((roles) => {
|
||||
module.exports.existingData.roles = roles.toJSON();
|
||||
|
||||
return models.User.findAll({columns: ['id', 'email']});
|
||||
})
|
||||
.then((users) => {
|
||||
module.exports.existingData.users = users.toJSON(context.internal);
|
||||
|
||||
return models.Tag.findAll({columns: ['id']});
|
||||
})
|
||||
.then((tags) => {
|
||||
module.exports.existingData.tags = tags.toJSON();
|
||||
|
||||
return models.ApiKey.findAll({withRelated: 'integration'});
|
||||
})
|
||||
.then((keys) => {
|
||||
module.exports.existingData.apiKeys = keys.toJSON(context.internal);
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* 1. reset & init db
|
||||
* 2. start the server once
|
||||
|
@ -166,149 +194,70 @@ const startGhost = async function startGhost(options) {
|
|||
// truncate database and re-run fixtures
|
||||
// we have to ensure that some components in Ghost are reloaded
|
||||
if (ghostServer && ghostServer.httpServer && !options.forceStart) {
|
||||
return dbUtils.teardown()
|
||||
.then(function () {
|
||||
return knexMigrator.init({only: 2});
|
||||
})
|
||||
.then(function () {
|
||||
settingsCache.shutdown();
|
||||
return settingsService.init();
|
||||
})
|
||||
.then(function () {
|
||||
return frontendSettingsService.init();
|
||||
})
|
||||
.then(function () {
|
||||
return themes.init();
|
||||
})
|
||||
.then(function () {
|
||||
urlServiceUtils.reset();
|
||||
return urlServiceUtils.isFinished();
|
||||
})
|
||||
.then(function () {
|
||||
web.shared.middlewares.customRedirects.reload();
|
||||
await dbUtils.teardown();
|
||||
|
||||
events.emit('server.start');
|
||||
await knexMigrator.init({only: 2});
|
||||
|
||||
/**
|
||||
* @TODO: this is dirty, but makes routing testing a lot easier for now, because the routing test
|
||||
* has no easy way to access existing resource id's, which are added from the Ghost fixtures.
|
||||
* I can do `testUtils.existingData.roles[0].id`.
|
||||
*/
|
||||
module.exports.existingData = {};
|
||||
return models.Role.findAll({columns: ['id']})
|
||||
.then((roles) => {
|
||||
module.exports.existingData.roles = roles.toJSON();
|
||||
settingsCache.shutdown();
|
||||
await settingsService.init();
|
||||
|
||||
return models.User.findAll({columns: ['id', 'email']});
|
||||
})
|
||||
.then((users) => {
|
||||
module.exports.existingData.users = users.toJSON(context.internal);
|
||||
await frontendSettingsService.init();
|
||||
await themes.init();
|
||||
|
||||
return models.Tag.findAll({columns: ['id']});
|
||||
})
|
||||
.then((tags) => {
|
||||
module.exports.existingData.tags = tags.toJSON();
|
||||
return models.ApiKey.findAll({withRelated: 'integration'});
|
||||
})
|
||||
.then((keys) => {
|
||||
module.exports.existingData.apiKeys = keys.toJSON(context.internal);
|
||||
console.timeEnd('Start Ghost'); // eslint-disable-line no-console
|
||||
})
|
||||
.return(ghostServer);
|
||||
});
|
||||
urlServiceUtils.reset();
|
||||
await urlServiceUtils.isFinished();
|
||||
web.shared.middlewares.customRedirects.reload();
|
||||
|
||||
events.emit('server.start');
|
||||
await dirtyDataFunction();
|
||||
console.log('Restart Mode'); // eslint-disable-line no-console
|
||||
console.timeEnd('Start Ghost'); // eslint-disable-line no-console
|
||||
|
||||
return ghostServer;
|
||||
}
|
||||
|
||||
return knexMigrator.reset({force: true})
|
||||
.then(function () {
|
||||
if (ghostServer && ghostServer.httpServer) {
|
||||
return ghostServer.stop();
|
||||
}
|
||||
})
|
||||
.then(function initialiseDatabase() {
|
||||
settingsCache.shutdown();
|
||||
settingsCache.reset();
|
||||
return knexMigrator.init();
|
||||
})
|
||||
.then(function setPragma() {
|
||||
if (config.get('database:client') === 'sqlite3') {
|
||||
return db.knex.raw('PRAGMA journal_mode = TRUNCATE;');
|
||||
} else {
|
||||
return Promise.resolve();
|
||||
}
|
||||
})
|
||||
.then(function initializeGhost() {
|
||||
urlService.resetGenerators();
|
||||
await knexMigrator.reset({force: true});
|
||||
|
||||
return ghost();
|
||||
})
|
||||
.then(function startGhostServer(_ghostServer) {
|
||||
ghostServer = _ghostServer;
|
||||
if (ghostServer && ghostServer.httpServer) {
|
||||
await ghostServer.stop();
|
||||
}
|
||||
|
||||
if (options.subdir) {
|
||||
parentApp = express('test parent');
|
||||
parentApp.use(urlUtils.getSubdir(), ghostServer.rootApp);
|
||||
return ghostServer.start(parentApp);
|
||||
}
|
||||
settingsCache.shutdown();
|
||||
settingsCache.reset();
|
||||
await knexMigrator.init();
|
||||
|
||||
return ghostServer.start();
|
||||
})
|
||||
.then(function () {
|
||||
let timeout;
|
||||
if (config.get('database:client') === 'sqlite3') {
|
||||
await db.knex.raw('PRAGMA journal_mode = TRUNCATE;');
|
||||
}
|
||||
|
||||
GhostServer.announceServerReadiness();
|
||||
urlService.resetGenerators();
|
||||
|
||||
return new Promise(function (resolve) {
|
||||
(function retry() {
|
||||
clearTimeout(timeout);
|
||||
ghostServer = await ghost();
|
||||
|
||||
if (urlService.hasFinished()) {
|
||||
return resolve();
|
||||
}
|
||||
if (options.subdir) {
|
||||
parentApp = express('test parent');
|
||||
parentApp.use(urlUtils.getSubdir(), ghostServer.rootApp);
|
||||
await ghostServer.start(parentApp);
|
||||
} else {
|
||||
await ghostServer.start();
|
||||
}
|
||||
|
||||
timeout = setTimeout(retry, 50);
|
||||
})();
|
||||
});
|
||||
})
|
||||
.then(function returnGhost() {
|
||||
/**
|
||||
* @TODO: this is dirty, but makes routing testing a lot easier for now, because the routing test
|
||||
* has no easy way to access existing resource id's, which are added from the Ghost fixtures.
|
||||
* I can do `testUtils.existingData.roles[0].id`.
|
||||
*/
|
||||
module.exports.existingData = {};
|
||||
return models.Role.findAll({columns: ['id']})
|
||||
.then((roles) => {
|
||||
module.exports.existingData.roles = roles.toJSON();
|
||||
GhostServer.announceServerReadiness();
|
||||
await urlServiceUtils.isFinished({disableDbReadyEvent: true});
|
||||
|
||||
return models.User.findAll({columns: ['id', 'email']});
|
||||
})
|
||||
.then((users) => {
|
||||
module.exports.existingData.users = users.toJSON(context.internal);
|
||||
|
||||
return models.Tag.findAll({columns: ['id']});
|
||||
})
|
||||
.then((tags) => {
|
||||
module.exports.existingData.tags = tags.toJSON();
|
||||
|
||||
return models.ApiKey.findAll({withRelated: 'integration'});
|
||||
})
|
||||
.then((keys) => {
|
||||
module.exports.existingData.apiKeys = keys.toJSON();
|
||||
console.timeEnd('Start Ghost'); // eslint-disable-line no-console
|
||||
})
|
||||
.return(ghostServer);
|
||||
});
|
||||
await dirtyDataFunction();
|
||||
console.log('Fresh Start Mode'); // eslint-disable-line no-console
|
||||
console.timeEnd('Start Ghost'); // eslint-disable-line no-console
|
||||
return ghostServer;
|
||||
};
|
||||
|
||||
module.exports = {
|
||||
startGhost: startGhost,
|
||||
|
||||
stopGhost: () => {
|
||||
stopGhost: async () => {
|
||||
if (ghostServer && ghostServer.httpServer) {
|
||||
return ghostServer.stop()
|
||||
.then(() => {
|
||||
urlService.resetGenerators();
|
||||
});
|
||||
await ghostServer.stop();
|
||||
urlService.resetGenerators();
|
||||
}
|
||||
},
|
||||
teardownDb: dbUtils.teardown,
|
||||
|
@ -351,7 +300,6 @@ module.exports = {
|
|||
initFixtures: initFixtures,
|
||||
initData: dbUtils.initData,
|
||||
clearData: dbUtils.clearData,
|
||||
clearBruteData: dbUtils.clearBruteData,
|
||||
setupRedirectsFile: redirects.setupFile,
|
||||
|
||||
fixtures: fixtureUtils.fixtures,
|
||||
|
|
Loading…
Reference in a new issue