const url = require('url'); const _ = require('lodash'); const testUtils = require('../../../../utils'); // NOTE: the dependance on the schema here is wrong! It is a design flaw which is causing problems for API maintenance and compatibility // whenever you need to modify any of the below property lists using schema - rework them into an "allowlist" array like it's done in // the commit introducing this comment. const schema = require('../../../../../core/server/data/schema').tables; const API_URL = '/ghost/api/canary/admin/'; const expectedProperties = { // API top level posts: ['posts', 'meta'], tags: ['tags', 'meta'], users: ['users', 'meta'], settings: ['settings', 'meta'], subscribers: ['subscribers', 'meta'], roles: ['roles'], pagination: ['page', 'limit', 'pages', 'total', 'next', 'prev'], slugs: ['slugs'], slug: ['slug'], invites: ['invites', 'meta'], themes: ['themes'], members: ['members', 'meta'], site: ['title', 'description', 'logo', 'icon', 'accent_color', 'url', 'version'], post: [ 'id', 'uuid', 'title', 'slug', 'mobiledoc', 'comment_id', 'feature_image', 'featured', 'status', 'visibility', 'email_recipient_filter', 'created_at', 'updated_at', 'published_at', 'custom_excerpt', 'codeinjection_head', 'codeinjection_foot', 'custom_template', 'canonical_url', 'url', 'primary_tag', 'primary_author', 'excerpt', 'tags', 'authors', 'email', 'og_image', 'og_title', 'og_description', 'twitter_image', 'twitter_title', 'twitter_description', 'meta_title', 'meta_description', 'email_subject', 'frontmatter' ], user: _(schema.users) .keys() .without('visibility') .without('password') .without('locale') .concat('url') , tag: _(schema.tags) .keys() // unused field .without('parent_id') , setting: _(schema.settings) .keys() , subscriber: _(schema.subscribers) .keys() , member: _(schema.members) .keys() .concat('avatar_image') .concat('comped') .concat('labels') , member_signin_url: ['member_id', 'url'], role: _(schema.roles) .keys() , permission: _(schema.permissions) .keys() , notification: ['type', 'message', 'status', 'id', 'dismissible', 'location', 'custom'], theme: ['name', 'package', 'active'], invite: _(schema.invites) .keys() .without('token') , webhook: _(schema.webhooks) .keys() , email_preview: ['html', 'subject', 'plaintext'] }; _.each(expectedProperties, (value, key) => { if (!value.__wrapped__) { return; } /** * @deprecated: x_by */ expectedProperties[key] = value .without( 'created_by', 'updated_by', 'published_by' ) .value(); }); module.exports = { API: { getApiQuery(route) { return url.resolve(API_URL, route); }, checkResponse(...args) { this.expectedProperties = expectedProperties; return testUtils.API.checkResponse.call(this, ...args); } }, doAuth(...args) { return testUtils.API.doAuth(`${API_URL}session/`, ...args); }, getValidAdminToken(endpoint, key) { const jwt = require('jsonwebtoken'); key = key || testUtils.DataGenerator.Content.api_keys[0]; const JWT_OPTIONS = { keyid: key.id, algorithm: 'HS256', expiresIn: '5m', audience: endpoint }; return jwt.sign( {}, Buffer.from(key.secret, 'hex'), JWT_OPTIONS ); } };