2013-05-16 12:21:13 +01:00
|
|
|
// # Ghost Data API
|
2014-06-03 14:05:25 +01:00
|
|
|
// Provides access from anywhere to the Ghost data layer.
|
|
|
|
//
|
|
|
|
// Ghost's JSON API is integral to the workings of Ghost, regardless of whether you want to access data internally,
|
|
|
|
// from a theme, an app, or from an external app, you'll use the Ghost JSON API to do so.
|
2013-05-16 12:21:13 +01:00
|
|
|
|
2018-09-26 13:13:41 +07:00
|
|
|
const {isEmpty} = require('lodash');
|
|
|
|
const Promise = require('bluebird');
|
|
|
|
const models = require('../models');
|
|
|
|
const urlService = require('../services/url');
|
|
|
|
const configuration = require('./configuration');
|
|
|
|
const db = require('./db');
|
|
|
|
const mail = require('./mail');
|
|
|
|
const notifications = require('./notifications');
|
|
|
|
const posts = require('./posts');
|
|
|
|
const schedules = require('./schedules');
|
|
|
|
const roles = require('./roles');
|
|
|
|
const settings = require('./settings');
|
|
|
|
const tags = require('./tags');
|
|
|
|
const invites = require('./invites');
|
|
|
|
const redirects = require('./redirects');
|
|
|
|
const clients = require('./clients');
|
|
|
|
const users = require('./users');
|
|
|
|
const slugs = require('./slugs');
|
|
|
|
const themes = require('./themes');
|
|
|
|
const subscribers = require('./subscribers');
|
|
|
|
const authentication = require('./authentication');
|
|
|
|
const uploads = require('./upload');
|
|
|
|
const exporter = require('../data/exporter');
|
|
|
|
const slack = require('./slack');
|
|
|
|
const webhooks = require('./webhooks');
|
|
|
|
const oembed = require('./oembed');
|
2013-06-25 12:43:15 +01:00
|
|
|
|
2017-03-20 18:02:44 +00:00
|
|
|
function isActiveThemeUpdate(method, endpoint, result) {
|
|
|
|
if (endpoint === 'themes') {
|
|
|
|
if (method === 'PUT') {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (method === 'POST' && result.themes && result.themes[0] && result.themes[0].active === true) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2016-09-14 11:18:52 +01:00
|
|
|
}
|
|
|
|
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
/**
|
|
|
|
* ### Cache Invalidation Header
|
|
|
|
* Calculate the header string for the X-Cache-Invalidate: header.
|
|
|
|
* The resulting string instructs any cache in front of the blog that request has occurred which invalidates any cached
|
|
|
|
* versions of the listed URIs.
|
|
|
|
*
|
|
|
|
* `/*` is used to mean the entire cache is invalid
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @param {Express.request} req Original HTTP Request
|
|
|
|
* @param {Object} result API method result
|
2014-12-28 05:27:29 +00:00
|
|
|
* @return {String} Resolves to header string
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
*/
|
2018-09-26 13:13:41 +07:00
|
|
|
const cacheInvalidationHeader = (req, result) => {
|
2018-09-10 18:00:48 +05:30
|
|
|
const parsedUrl = req._parsedUrl.pathname.replace(/^\/|\/$/g, '').split('/'),
|
2013-09-24 17:21:43 +02:00
|
|
|
method = req.method,
|
2014-07-08 23:28:31 +01:00
|
|
|
endpoint = parsedUrl[0],
|
2016-05-19 13:49:22 +02:00
|
|
|
subdir = parsedUrl[1],
|
2014-04-21 20:04:30 -05:00
|
|
|
jsonResult = result.toJSON ? result.toJSON() : result,
|
2018-09-10 18:00:48 +05:30
|
|
|
INVALIDATE_ALL = '/*';
|
|
|
|
|
|
|
|
let post,
|
2014-06-03 22:38:15 +00:00
|
|
|
hasStatusChanged,
|
|
|
|
wasPublishedUpdated;
|
2013-11-03 18:13:19 +01:00
|
|
|
|
2017-03-20 18:02:44 +00:00
|
|
|
if (isActiveThemeUpdate(method, endpoint, result)) {
|
2016-09-14 11:18:52 +01:00
|
|
|
// Special case for if we're overwriting an active theme
|
|
|
|
return INVALIDATE_ALL;
|
|
|
|
} else if (['POST', 'PUT', 'DELETE'].indexOf(method) > -1) {
|
2016-05-19 13:49:22 +02:00
|
|
|
if (endpoint === 'schedules' && subdir === 'posts') {
|
|
|
|
return INVALIDATE_ALL;
|
|
|
|
}
|
2017-09-27 19:58:33 +02:00
|
|
|
if (['settings', 'users', 'db', 'tags', 'redirects'].indexOf(endpoint) > -1) {
|
2016-03-20 12:26:42 -05:00
|
|
|
return INVALIDATE_ALL;
|
2013-09-24 17:21:43 +02:00
|
|
|
} else if (endpoint === 'posts') {
|
2016-03-20 12:26:42 -05:00
|
|
|
if (method === 'DELETE') {
|
|
|
|
return INVALIDATE_ALL;
|
|
|
|
}
|
|
|
|
|
2014-04-21 20:04:30 -05:00
|
|
|
post = jsonResult.posts[0];
|
2014-06-03 22:38:15 +00:00
|
|
|
hasStatusChanged = post.statusChanged;
|
|
|
|
// Invalidate cache when post was updated but not when post is draft
|
|
|
|
wasPublishedUpdated = method === 'PUT' && post.status === 'published';
|
2014-04-21 20:04:30 -05:00
|
|
|
|
|
|
|
// Remove the statusChanged value from the response
|
2014-06-03 22:38:15 +00:00
|
|
|
delete post.statusChanged;
|
2014-04-21 20:04:30 -05:00
|
|
|
|
|
|
|
// Don't set x-cache-invalidate header for drafts
|
2016-03-20 12:26:42 -05:00
|
|
|
if (hasStatusChanged || wasPublishedUpdated) {
|
|
|
|
return INVALIDATE_ALL;
|
2015-05-18 20:56:41 +01:00
|
|
|
} else {
|
2018-04-17 17:36:05 +08:00
|
|
|
// routeKeywords.preview: 'p'
|
|
|
|
return urlService.utils.urlFor({relativeUrl: urlService.utils.urlJoin('/p', post.uuid, '/')});
|
2013-09-24 17:21:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ### Location Header
|
|
|
|
*
|
|
|
|
* If the API request results in the creation of a new object, construct a Location: header which points to the new
|
|
|
|
* resource.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @param {Express.request} req Original HTTP Request
|
|
|
|
* @param {Object} result API method result
|
2014-12-28 05:27:29 +00:00
|
|
|
* @return {String} Resolves to header string
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
*/
|
2018-09-26 13:13:41 +07:00
|
|
|
const locationHeader = (req, result) => {
|
2018-09-25 01:19:21 +05:30
|
|
|
const apiRoot = urlService.utils.urlFor('api', {version: 'stable'});
|
2018-09-10 18:00:48 +05:30
|
|
|
let location,
|
2016-11-14 21:38:55 +07:00
|
|
|
newObject,
|
|
|
|
statusQuery;
|
2014-05-01 23:42:23 +00:00
|
|
|
|
|
|
|
if (req.method === 'POST') {
|
|
|
|
if (result.hasOwnProperty('posts')) {
|
2014-07-23 06:13:20 +00:00
|
|
|
newObject = result.posts[0];
|
2018-09-10 18:00:48 +05:30
|
|
|
statusQuery = `/?status=${newObject.status}`;
|
2017-12-11 19:14:05 +01:00
|
|
|
location = urlService.utils.urlJoin(apiRoot, 'posts', newObject.id, statusQuery);
|
2014-07-23 06:13:20 +00:00
|
|
|
} else if (result.hasOwnProperty('notifications')) {
|
|
|
|
newObject = result.notifications[0];
|
2017-12-11 19:14:05 +01:00
|
|
|
location = urlService.utils.urlJoin(apiRoot, 'notifications', newObject.id, '/');
|
2014-07-23 06:13:20 +00:00
|
|
|
} else if (result.hasOwnProperty('users')) {
|
|
|
|
newObject = result.users[0];
|
2017-12-11 19:14:05 +01:00
|
|
|
location = urlService.utils.urlJoin(apiRoot, 'users', newObject.id, '/');
|
2014-12-28 05:27:29 +00:00
|
|
|
} else if (result.hasOwnProperty('tags')) {
|
|
|
|
newObject = result.tags[0];
|
2017-12-11 19:14:05 +01:00
|
|
|
location = urlService.utils.urlJoin(apiRoot, 'tags', newObject.id, '/');
|
2017-11-21 15:43:14 +00:00
|
|
|
} else if (result.hasOwnProperty('webhooks')) {
|
|
|
|
newObject = result.webhooks[0];
|
2017-12-11 19:14:05 +01:00
|
|
|
location = urlService.utils.urlJoin(apiRoot, 'webhooks', newObject.id, '/');
|
2014-05-01 23:42:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-28 05:27:29 +00:00
|
|
|
return location;
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ### 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.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @see http://tools.ietf.org/html/rfc598
|
|
|
|
* @return {string}
|
|
|
|
*/
|
2016-04-14 22:44:05 +02:00
|
|
|
|
2018-09-26 13:13:41 +07:00
|
|
|
const contentDispositionHeaderExport = () => {
|
2018-09-18 19:29:56 +05:30
|
|
|
return exporter.fileName().then((filename) => {
|
2018-09-10 18:00:48 +05:30
|
|
|
return `Attachment; filename="${filename}"`;
|
2014-07-19 13:20:16 +01:00
|
|
|
});
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
};
|
|
|
|
|
2018-09-26 13:13:41 +07:00
|
|
|
const contentDispositionHeaderSubscribers = () => {
|
2018-09-10 18:00:48 +05:30
|
|
|
const datetime = (new Date()).toJSON().substring(0, 10);
|
|
|
|
return Promise.resolve(`Attachment; filename="subscribers.${datetime}.csv"`);
|
2016-04-14 22:44:05 +02:00
|
|
|
};
|
|
|
|
|
2018-09-26 13:13:41 +07:00
|
|
|
const contentDispositionHeaderRedirects = () => {
|
2017-09-21 17:01:03 +02:00
|
|
|
return Promise.resolve('Attachment; filename="redirects.json"');
|
|
|
|
};
|
|
|
|
|
2018-09-26 13:13:41 +07:00
|
|
|
const contentDispositionHeaderRoutes = () => {
|
2018-07-12 13:40:37 +02:00
|
|
|
return Promise.resolve('Attachment; filename="routes.yaml"');
|
|
|
|
};
|
|
|
|
|
2018-09-26 13:13:41 +07:00
|
|
|
const addHeaders = (apiMethod, req, res, result) => {
|
2018-09-10 18:00:48 +05:30
|
|
|
let cacheInvalidation,
|
2014-07-19 13:20:16 +01:00
|
|
|
location,
|
|
|
|
contentDisposition;
|
|
|
|
|
2014-12-28 05:27:29 +00:00
|
|
|
cacheInvalidation = cacheInvalidationHeader(req, result);
|
|
|
|
if (cacheInvalidation) {
|
|
|
|
res.set({'X-Cache-Invalidate': cacheInvalidation});
|
|
|
|
}
|
2014-07-19 13:20:16 +01:00
|
|
|
|
|
|
|
if (req.method === 'POST') {
|
2014-12-28 05:27:29 +00:00
|
|
|
location = locationHeader(req, result);
|
|
|
|
if (location) {
|
|
|
|
res.set({Location: location});
|
|
|
|
// The location header indicates that a new object was created.
|
|
|
|
// In this case the status code should be 201 Created
|
|
|
|
res.status(201);
|
|
|
|
}
|
2014-07-19 13:20:16 +01:00
|
|
|
}
|
|
|
|
|
2016-04-14 22:44:05 +02:00
|
|
|
// Add Export Content-Disposition Header
|
2014-07-19 13:20:16 +01:00
|
|
|
if (apiMethod === db.exportContent) {
|
2016-04-14 22:44:05 +02:00
|
|
|
contentDisposition = contentDispositionHeaderExport()
|
2018-09-18 19:29:56 +05:30
|
|
|
.then((header) => {
|
2016-04-14 22:44:05 +02:00
|
|
|
res.set({
|
|
|
|
'Content-Disposition': header
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add Subscribers Content-Disposition Header
|
|
|
|
if (apiMethod === subscribers.exportCSV) {
|
|
|
|
contentDisposition = contentDispositionHeaderSubscribers()
|
2018-09-18 19:29:56 +05:30
|
|
|
.then((header) => {
|
2016-04-14 22:44:05 +02:00
|
|
|
res.set({
|
|
|
|
'Content-Disposition': header,
|
|
|
|
'Content-Type': 'text/csv'
|
|
|
|
});
|
2014-07-19 13:20:16 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-09-21 17:01:03 +02:00
|
|
|
// Add Redirects Content-Disposition Header
|
|
|
|
if (apiMethod === redirects.download) {
|
|
|
|
contentDisposition = contentDispositionHeaderRedirects()
|
2018-09-18 19:29:56 +05:30
|
|
|
.then((header) => {
|
2017-09-21 17:01:03 +02:00
|
|
|
res.set({
|
|
|
|
'Content-Disposition': header,
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
'Content-Length': JSON.stringify(result).length
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-07-12 13:40:37 +02:00
|
|
|
// Add Routes Content-Disposition Header
|
|
|
|
if (apiMethod === settings.download) {
|
|
|
|
contentDisposition = contentDispositionHeaderRoutes()
|
|
|
|
.then((header) => {
|
|
|
|
res.set({
|
|
|
|
'Content-Disposition': header,
|
|
|
|
'Content-Type': 'application/yaml',
|
|
|
|
'Content-Length': JSON.stringify(result).length
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-12-28 05:27:29 +00:00
|
|
|
return contentDisposition;
|
2014-07-19 13:20:16 +01:00
|
|
|
};
|
|
|
|
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
/**
|
|
|
|
* ### HTTP
|
|
|
|
*
|
|
|
|
* Decorator for API functions which are called via an HTTP request. Takes the API method and wraps it so that it gets
|
|
|
|
* data from the request and returns a sensible JSON response.
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
* @param {Function} apiMethod API method to call
|
|
|
|
* @return {Function} middleware format function to be called by the route when a matching request is made
|
|
|
|
*/
|
2018-09-26 13:13:41 +07:00
|
|
|
const http = (apiMethod) => {
|
2015-06-14 20:07:52 +01:00
|
|
|
return function apiHandler(req, res, next) {
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
// We define 2 properties for using as arguments in API calls:
|
2018-08-30 18:30:36 +02:00
|
|
|
let object = req.body,
|
2018-09-10 18:00:48 +05:30
|
|
|
options = Object.assign({}, req.file, {ip: req.ip}, req.query, req.params, {
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
context: {
|
2018-08-03 15:14:38 +02:00
|
|
|
// @TODO: forward the client and user obj (options.context.user.id)
|
2016-11-09 16:01:07 +01:00
|
|
|
user: ((req.user && req.user.id) || (req.user && models.User.isExternalUser(req.user.id))) ? req.user.id : null,
|
2016-09-30 13:45:59 +02:00
|
|
|
client: (req.client && req.client.slug) ? req.client.slug : null,
|
|
|
|
client_id: (req.client && req.client.id) ? req.client.id : null
|
2014-05-07 01:28:51 -04:00
|
|
|
}
|
2014-01-03 00:37:21 +00:00
|
|
|
});
|
2014-05-05 15:51:21 +02:00
|
|
|
|
2018-08-30 18:30:36 +02:00
|
|
|
if (req.files) {
|
|
|
|
options.files = req.files;
|
|
|
|
}
|
|
|
|
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
// If this is a GET, or a DELETE, req.body should be null, so we only have options (route and query params)
|
|
|
|
// If this is a PUT, POST, or PATCH, req.body is an object
|
2018-09-10 18:00:48 +05:30
|
|
|
if (isEmpty(object)) {
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
object = options;
|
|
|
|
options = {};
|
|
|
|
}
|
2014-05-07 01:28:51 -04:00
|
|
|
|
2018-09-18 19:29:56 +05:30
|
|
|
return apiMethod(object, options).tap((response) => {
|
2014-12-28 05:27:29 +00:00
|
|
|
// Add X-Cache-Invalidate, Location, and Content-Disposition headers
|
2015-05-28 08:58:52 -05:00
|
|
|
return addHeaders(apiMethod, req, res, (response || {}));
|
2018-09-18 19:29:56 +05:30
|
|
|
}).then((response) => {
|
2018-09-26 13:13:41 +07:00
|
|
|
// CASE: api method response wants to handle the express response
|
|
|
|
// example: serve files (stream)
|
|
|
|
if (typeof response === 'function') {
|
|
|
|
return response(req, res, next);
|
|
|
|
}
|
|
|
|
|
2016-03-20 12:26:42 -05:00
|
|
|
if (req.method === 'DELETE') {
|
|
|
|
return res.status(204).end();
|
|
|
|
}
|
2018-07-12 13:40:37 +02:00
|
|
|
|
|
|
|
// Keep CSV, yaml formatting
|
|
|
|
if (res.get('Content-Type') && res.get('Content-Type').indexOf('text/csv') === 0 ||
|
|
|
|
res.get('Content-Type') && res.get('Content-Type').indexOf('application/yaml') === 0) {
|
2016-04-14 22:44:05 +02:00
|
|
|
return res.status(200).send(response);
|
|
|
|
}
|
2016-08-23 14:07:25 +02:00
|
|
|
|
2014-12-28 05:27:29 +00:00
|
|
|
// Send a properly formatting HTTP response containing the data with correct headers
|
|
|
|
res.json(response || {});
|
2018-09-18 19:29:56 +05:30
|
|
|
}).catch((error) => {
|
2015-06-14 20:07:52 +01:00
|
|
|
// To be handled by the API middleware
|
|
|
|
next(error);
|
2014-12-28 05:27:29 +00:00
|
|
|
});
|
2013-06-08 02:05:40 -03:00
|
|
|
};
|
2013-06-25 12:43:15 +01:00
|
|
|
};
|
|
|
|
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
/**
|
|
|
|
* ## Public API
|
|
|
|
*/
|
2013-12-06 09:51:35 +01:00
|
|
|
module.exports = {
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
http: http,
|
|
|
|
// API Endpoints
|
2014-08-20 21:42:34 +00:00
|
|
|
configuration: configuration,
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
db: db,
|
|
|
|
mail: mail,
|
|
|
|
notifications: notifications,
|
2013-12-06 09:51:35 +01:00
|
|
|
posts: posts,
|
2016-05-19 13:49:22 +02:00
|
|
|
schedules: schedules,
|
2014-07-15 16:22:06 +01:00
|
|
|
roles: roles,
|
Refactor API arguments
closes #2610, refs #2697
- cleanup API index.js, and add docs
- all API methods take consistent arguments: object & options
- browse, read, destroy take options, edit and add take object and options
- the context is passed as part of options, meaning no more .call
everywhere
- destroy expects an object, rather than an id all the way down to the model layer
- route params such as :id, :slug, and :key are passed as an option & used
to perform reads, updates and deletes where possible - settings / themes
may need work here still
- HTTP posts api can find a post by slug
- Add API utils for checkData
2014-05-08 13:41:19 +01:00
|
|
|
settings: settings,
|
2013-12-06 09:51:35 +01:00
|
|
|
tags: tags,
|
2015-08-31 15:59:56 +02:00
|
|
|
clients: clients,
|
2014-05-06 00:38:05 +02:00
|
|
|
users: users,
|
2014-06-20 11:15:01 +02:00
|
|
|
slugs: slugs,
|
2016-04-14 22:44:05 +02:00
|
|
|
subscribers: subscribers,
|
2014-07-15 12:40:14 +02:00
|
|
|
authentication: authentication,
|
2016-03-29 10:40:44 +02:00
|
|
|
uploads: uploads,
|
2016-08-23 14:07:25 +02:00
|
|
|
slack: slack,
|
2016-09-21 16:48:14 +02:00
|
|
|
themes: themes,
|
2017-09-21 17:01:03 +02:00
|
|
|
invites: invites,
|
2017-11-21 15:43:14 +00:00
|
|
|
redirects: redirects,
|
2018-06-05 15:49:23 +01:00
|
|
|
webhooks: webhooks,
|
|
|
|
oembed: oembed
|
2013-12-06 09:51:35 +01:00
|
|
|
};
|
2014-06-03 14:05:25 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* ## API Methods
|
|
|
|
*
|
|
|
|
* Most API methods follow the BREAD pattern, although not all BREAD methods are available for all resources.
|
|
|
|
* Most API methods have a similar signature, they either take just `options`, or both `object` and `options`.
|
|
|
|
* For RESTful resources `object` is always a model object of the correct type in the form `name: [{object}]`
|
|
|
|
* `options` is an object with several named properties, the possibilities are listed for each method.
|
|
|
|
*
|
|
|
|
* Read / Edit / Destroy routes expect some sort of identifier (id / slug / key) for which object they are handling
|
|
|
|
*
|
|
|
|
* All API methods take a context object as one of the options:
|
|
|
|
*
|
|
|
|
* @typedef context
|
|
|
|
* Context provides information for determining permissions. Usually a user, but sometimes an app, or the internal flag
|
|
|
|
* @param {Number} user (optional)
|
|
|
|
* @param {String} app (optional)
|
|
|
|
* @param {Boolean} internal (optional)
|
|
|
|
*/
|