mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-04 17:04:59 +03:00
cbdc91ce48
refs #2635 - Adds 'Location' header to endpoints which create new resources and have corresponding `GET` endpoint as speced in JSON API - https://jsonapi.org/format/#crud-creating-responses-201. Specifically: /posts/ /pages/ /integrations/ /tags/ /members/ /labels/ /notifications/ /invites/ - Adding the header should allow for better resource discoverability and improved logging readability - Added `url` property to the frame constructor. Data in `url` should give enough information to later build up the `Location` header URL for created resource. - Added Location header to headers handler. The Location value is built up from a combination of request URL and the id that is present in the response for the resource. The header is automatically added to requests coming to `add` controller methods which return `id` property in the frame result - Excluded Webhooks API as there is no "GET" endpoint available to fetch the resource
153 lines
4.5 KiB
JavaScript
153 lines
4.5 KiB
JavaScript
const url = require('url');
|
|
const debug = require('ghost-ignition').debug('api:shared:headers');
|
|
const Promise = require('bluebird');
|
|
const INVALIDATE_ALL = '/*';
|
|
|
|
const cacheInvalidate = (result, options = {}) => {
|
|
let value = options.value;
|
|
|
|
return {
|
|
'X-Cache-Invalidate': value || INVALIDATE_ALL
|
|
};
|
|
};
|
|
|
|
const disposition = {
|
|
/**
|
|
* @description Generate CSV header.
|
|
*
|
|
* @param {Object} result - API response
|
|
* @param {Object} options
|
|
* @return {Object}
|
|
*/
|
|
csv(result, options = {}) {
|
|
let value = options.value;
|
|
|
|
if (typeof options.value === 'function') {
|
|
value = options.value();
|
|
}
|
|
|
|
return {
|
|
'Content-Disposition': `Attachment; filename="${value}"`,
|
|
'Content-Type': 'text/csv'
|
|
};
|
|
},
|
|
|
|
/**
|
|
* @description Generate JSON header.
|
|
*
|
|
* @param {Object} result - API response
|
|
* @param {Object} options
|
|
* @return {Object}
|
|
*/
|
|
json(result, options = {}) {
|
|
return {
|
|
'Content-Disposition': `Attachment; filename="${options.value}"`,
|
|
'Content-Type': 'application/json',
|
|
'Content-Length': Buffer.byteLength(JSON.stringify(result))
|
|
};
|
|
},
|
|
|
|
/**
|
|
* @description Generate YAML header.
|
|
*
|
|
* @param {Object} result - API response
|
|
* @param {Object} options
|
|
* @return {Object}
|
|
*/
|
|
yaml(result, options = {}) {
|
|
return {
|
|
'Content-Disposition': `Attachment; filename="${options.value}"`,
|
|
'Content-Type': 'application/yaml',
|
|
'Content-Length': Buffer.byteLength(JSON.stringify(result))
|
|
};
|
|
},
|
|
|
|
/**
|
|
* @description Content Disposition Header
|
|
*
|
|
* Create a header that invokes the 'Save As' dialog in the browser when exporting the database to file. The 'filename'
|
|
* parameter is governed by [RFC6266](http://tools.ietf.org/html/rfc6266#section-4.3).
|
|
*
|
|
* For encoding whitespace and non-ISO-8859-1 characters, you MUST use the "filename*=" attribute, NOT "filename=".
|
|
* Ideally, both. Examples: http://tools.ietf.org/html/rfc6266#section-5
|
|
*
|
|
* We'll use ISO-8859-1 characters here to keep it simple.
|
|
*
|
|
* @see http://tools.ietf.org/html/rfc598
|
|
*/
|
|
file(result, options = {}) {
|
|
return Promise.resolve()
|
|
.then(() => {
|
|
let value = options.value;
|
|
|
|
if (typeof options.value === 'function') {
|
|
value = options.value();
|
|
}
|
|
|
|
return value;
|
|
})
|
|
.then((filename) => {
|
|
return {
|
|
'Content-Disposition': `Attachment; filename="${filename}"`
|
|
};
|
|
});
|
|
}
|
|
};
|
|
|
|
module.exports = {
|
|
/**
|
|
* @description Get header based on ctrl configuration.
|
|
*
|
|
* @param {Object} result - API response
|
|
* @param {Object} apiConfigHeaders
|
|
* @param {Object} frame
|
|
* @return {Promise}
|
|
*/
|
|
async get(result, apiConfigHeaders = {}, frame) {
|
|
let headers = {};
|
|
|
|
if (apiConfigHeaders.disposition) {
|
|
const dispositionHeader = await disposition[apiConfigHeaders.disposition.type](result, apiConfigHeaders.disposition);
|
|
|
|
if (dispositionHeader) {
|
|
Object.assign(headers, dispositionHeader);
|
|
}
|
|
}
|
|
|
|
if (apiConfigHeaders.cacheInvalidate) {
|
|
const cacheInvalidationHeader = cacheInvalidate(result, apiConfigHeaders.cacheInvalidate);
|
|
|
|
if (cacheInvalidationHeader) {
|
|
Object.assign(headers, cacheInvalidationHeader);
|
|
}
|
|
}
|
|
|
|
const locationHeaderDisabled = apiConfigHeaders && apiConfigHeaders.location === false;
|
|
const hasFrameData = frame
|
|
&& (frame.method === 'add')
|
|
&& result[frame.docName]
|
|
&& result[frame.docName][0]
|
|
&& result[frame.docName][0].id;
|
|
|
|
if (!locationHeaderDisabled && hasFrameData) {
|
|
const protocol = (frame.original.url.secure === false) ? 'http://' : 'https://';
|
|
const resourceId = result[frame.docName][0].id;
|
|
|
|
let locationURL = url.resolve(`${protocol}${frame.original.url.host}`,frame.original.url.pathname);
|
|
if (!locationURL.endsWith('/')) {
|
|
locationURL += '/';
|
|
}
|
|
locationURL += `${resourceId}/`;
|
|
|
|
const locationHeader = {
|
|
Location: locationURL
|
|
};
|
|
|
|
Object.assign(headers, locationHeader);
|
|
}
|
|
|
|
debug(headers);
|
|
return headers;
|
|
}
|
|
};
|