2020-05-25 03:49:38 -05:00
|
|
|
const errors = require('@tryghost/errors');
|
2018-12-17 16:14:36 +01:00
|
|
|
const _ = require('lodash');
|
|
|
|
const url = require('url');
|
|
|
|
const moment = require('moment');
|
|
|
|
const DataGenerator = require('./fixtures/data-generator');
|
2020-05-27 12:47:53 -05:00
|
|
|
const config = require('../../core/shared/config');
|
2020-08-11 18:44:21 +01:00
|
|
|
const {sequence} = require('@tryghost/promise');
|
2018-12-17 16:14:36 +01:00
|
|
|
const host = config.get('server').host;
|
|
|
|
const port = config.get('server').port;
|
|
|
|
const protocol = 'http://';
|
2013-11-07 10:34:18 +01:00
|
|
|
|
2016-02-14 12:46:09 +00:00
|
|
|
function getURL() {
|
|
|
|
return protocol + host;
|
|
|
|
}
|
|
|
|
|
2014-04-28 21:42:38 +01:00
|
|
|
function getSigninURL() {
|
2016-02-14 12:46:09 +00:00
|
|
|
return url.resolve(protocol + host + ':' + port, 'ghost/signin/');
|
2013-10-07 20:39:33 -05:00
|
|
|
}
|
2016-02-14 12:46:09 +00:00
|
|
|
|
2014-04-28 21:42:38 +01:00
|
|
|
function getAdminURL() {
|
2016-02-14 12:46:09 +00:00
|
|
|
return url.resolve(protocol + host + ':' + port, 'ghost/');
|
|
|
|
}
|
|
|
|
|
|
|
|
function isISO8601(date) {
|
|
|
|
return moment(date).parsingFlags().iso;
|
2013-11-24 15:29:36 +01:00
|
|
|
}
|
2013-10-07 20:39:33 -05:00
|
|
|
|
2013-11-03 18:13:19 +01:00
|
|
|
// make sure the API only returns expected properties only
|
2016-02-14 12:46:09 +00:00
|
|
|
function checkResponseValue(jsonResponse, expectedProperties) {
|
2020-04-29 16:44:27 +01:00
|
|
|
const providedProperties = _.keys(jsonResponse);
|
|
|
|
const missing = _.difference(expectedProperties, providedProperties);
|
|
|
|
const unexpected = _.difference(providedProperties, expectedProperties);
|
2016-02-14 12:46:09 +00:00
|
|
|
|
|
|
|
_.each(missing, function (prop) {
|
|
|
|
jsonResponse.should.have.property(prop);
|
|
|
|
});
|
|
|
|
|
|
|
|
_.each(unexpected, function (prop) {
|
|
|
|
jsonResponse.should.not.have.property(prop);
|
|
|
|
});
|
|
|
|
|
|
|
|
providedProperties.length.should.eql(expectedProperties.length);
|
2013-10-07 20:39:33 -05:00
|
|
|
}
|
|
|
|
|
2017-09-28 15:00:52 +02:00
|
|
|
// @TODO: support options pattern only, it's annoying to call checkResponse(null, null, null, something)
|
|
|
|
function checkResponse(jsonResponse, objectType, additionalProperties, missingProperties, onlyProperties, options) {
|
|
|
|
options = options || {};
|
|
|
|
|
2018-12-17 16:14:36 +01:00
|
|
|
let checkProperties = options.public ? (this.expectedProperties[objectType].public || this.expectedProperties[objectType]) : (this.expectedProperties[objectType].default || this.expectedProperties[objectType]);
|
2017-05-30 11:40:39 +01:00
|
|
|
|
|
|
|
checkProperties = onlyProperties ? onlyProperties : checkProperties;
|
2014-07-08 18:00:59 +02:00
|
|
|
checkProperties = additionalProperties ? checkProperties.concat(additionalProperties) : checkProperties;
|
2015-04-17 22:27:04 +01:00
|
|
|
checkProperties = missingProperties ? _.xor(checkProperties, missingProperties) : checkProperties;
|
2014-07-08 18:00:59 +02:00
|
|
|
|
|
|
|
checkResponseValue(jsonResponse, checkProperties);
|
2014-04-28 21:42:38 +01:00
|
|
|
}
|
|
|
|
|
2018-10-05 15:08:30 +02:00
|
|
|
/**
|
|
|
|
* This function manages the work of ensuring we have an overridden owner user, and grabbing an access token
|
|
|
|
*
|
|
|
|
* @TODO make this do the DB init as well
|
|
|
|
*/
|
|
|
|
const doAuth = (apiOptions) => {
|
|
|
|
return function doAuthInner() {
|
|
|
|
let API_URL = arguments[0];
|
|
|
|
let request = arguments[1];
|
|
|
|
|
|
|
|
// Remove API_URL & request from this list
|
2018-10-05 16:17:34 +02:00
|
|
|
let options = Array.prototype.slice.call(arguments, 2);
|
2018-10-05 15:08:30 +02:00
|
|
|
|
|
|
|
// No DB setup, but override the owner
|
|
|
|
options = _.merge({'owner:post': true}, _.transform(options, function (result, val) {
|
|
|
|
if (val) {
|
|
|
|
result[val] = true;
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
2018-10-05 16:17:34 +02:00
|
|
|
const fixtureOps = apiOptions.getFixtureOps(options);
|
2018-10-05 15:08:30 +02:00
|
|
|
|
|
|
|
return sequence(fixtureOps).then(function () {
|
|
|
|
return login(request, API_URL);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
const login = (request, API_URL) => {
|
|
|
|
// CASE: by default we use the owner to login
|
|
|
|
if (!request.user) {
|
|
|
|
request.user = DataGenerator.Content.users[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise(function (resolve, reject) {
|
|
|
|
request.post(API_URL)
|
|
|
|
.set('Origin', config.get('url'))
|
|
|
|
.send({
|
|
|
|
grant_type: 'password',
|
|
|
|
username: request.user.email,
|
2019-09-12 14:54:39 +01:00
|
|
|
password: 'Sl1m3rson99'
|
2018-10-09 10:24:41 +07:00
|
|
|
})
|
|
|
|
.then(function then(res) {
|
2019-11-06 12:48:58 +07:00
|
|
|
if (res.statusCode === 302) {
|
|
|
|
// This can happen if you already have an instance running e.g. if you've been using Ghost CLI recently
|
2021-12-01 10:22:01 +00:00
|
|
|
return reject(new errors.InternalServerError({
|
2019-11-06 12:48:58 +07:00
|
|
|
message: 'Ghost is redirecting, do you have an instance already running on port 2369?'
|
|
|
|
}));
|
|
|
|
} else if (res.statusCode !== 200 && res.statusCode !== 201) {
|
2021-12-01 10:22:01 +00:00
|
|
|
return reject(new errors.InternalServerError({
|
2021-12-09 13:52:02 +04:00
|
|
|
message: _.get(res, 'body.errors[0].message') || res.error.message
|
2018-10-09 10:24:41 +07:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2019-09-11 11:28:55 +02:00
|
|
|
resolve(res.headers['set-cookie']);
|
2018-10-09 10:24:41 +07:00
|
|
|
}, reject);
|
2018-10-05 15:08:30 +02:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = (options = {}) => {
|
|
|
|
return {
|
|
|
|
getSigninURL: getSigninURL,
|
|
|
|
getAdminURL: getAdminURL,
|
|
|
|
doAuth: doAuth(options),
|
|
|
|
login: login,
|
|
|
|
getURL: getURL,
|
|
|
|
checkResponse: checkResponse,
|
|
|
|
checkResponseValue: checkResponseValue,
|
|
|
|
isISO8601: isISO8601
|
|
|
|
};
|
2013-11-07 10:34:18 +01:00
|
|
|
};
|