2013-05-16 12:21:13 +01:00
|
|
|
// # Ghost Data API
|
|
|
|
// Provides access to the data model
|
|
|
|
|
2014-02-05 00:40:30 -08:00
|
|
|
var _ = require('lodash'),
|
2013-12-06 09:51:35 +01:00
|
|
|
when = require('when'),
|
2013-12-30 02:03:29 -05:00
|
|
|
config = require('../config'),
|
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
|
|
|
// Include Endpoints
|
2013-12-06 09:51:35 +01:00
|
|
|
db = require('./db'),
|
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
|
|
|
mail = require('./mail'),
|
2013-12-06 09:51:35 +01:00
|
|
|
notifications = require('./notifications'),
|
|
|
|
posts = require('./posts'),
|
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 = require('./settings'),
|
2013-12-06 09:51:35 +01:00
|
|
|
tags = require('./tags'),
|
2014-05-13 14:33:34 +00:00
|
|
|
themes = require('./themes'),
|
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
|
|
|
users = require('./users'),
|
2014-05-06 00:38:05 +02:00
|
|
|
slugs = require('./slugs'),
|
2013-08-30 13:20:30 +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
|
|
|
http,
|
|
|
|
formatHttpErrors,
|
|
|
|
cacheInvalidationHeader,
|
|
|
|
locationHeader,
|
|
|
|
contentDispositionHeader,
|
2014-05-06 00:38:05 +02:00
|
|
|
init;
|
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
|
|
|
|
|
|
|
/**
|
|
|
|
* ### Init
|
|
|
|
* Initialise the API - populate the settings cache
|
|
|
|
* @return {Promise(Settings)} Resolves to Settings Collection
|
|
|
|
*/
|
|
|
|
init = function () {
|
|
|
|
return settings.updateSettingsCache();
|
|
|
|
};
|
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
|
|
|
/**
|
|
|
|
* ### 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
|
|
|
|
* @return {Promise(String)} Resolves to header string
|
|
|
|
*/
|
|
|
|
cacheInvalidationHeader = function (req, result) {
|
2013-09-24 17:21:43 +02:00
|
|
|
var parsedUrl = req._parsedUrl.pathname.replace(/\/$/, '').split('/'),
|
|
|
|
method = req.method,
|
2013-11-03 18:13:19 +01:00
|
|
|
endpoint = parsedUrl[4],
|
|
|
|
id = parsedUrl[5],
|
|
|
|
cacheInvalidate,
|
2014-04-21 20:04:30 -05:00
|
|
|
jsonResult = result.toJSON ? result.toJSON() : result,
|
|
|
|
post,
|
|
|
|
wasPublished,
|
|
|
|
wasDeleted;
|
2013-11-03 18:13:19 +01:00
|
|
|
|
2013-09-24 17:21:43 +02:00
|
|
|
if (method === 'POST' || method === 'PUT' || method === 'DELETE') {
|
2013-12-25 01:05:20 +01:00
|
|
|
if (endpoint === 'settings' || endpoint === 'users' || endpoint === 'db') {
|
2014-03-26 12:45:54 +00:00
|
|
|
cacheInvalidate = '/*';
|
2013-09-24 17:21:43 +02:00
|
|
|
} else if (endpoint === 'posts') {
|
2014-04-21 20:04:30 -05:00
|
|
|
post = jsonResult.posts[0];
|
|
|
|
wasPublished = post.statusChanged && post.status === 'published';
|
|
|
|
wasDeleted = method === 'DELETE';
|
|
|
|
|
|
|
|
// Remove the statusChanged value from the response
|
|
|
|
if (post.statusChanged) {
|
|
|
|
delete post.statusChanged;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't set x-cache-invalidate header for drafts
|
|
|
|
if (wasPublished || wasDeleted) {
|
|
|
|
cacheInvalidate = '/, /page/*, /rss/, /rss/*, /tag/*';
|
|
|
|
if (id && post.slug) {
|
|
|
|
return config.urlForPost(settings, post).then(function (postUrl) {
|
|
|
|
return cacheInvalidate + ', ' + postUrl;
|
|
|
|
});
|
|
|
|
}
|
2013-09-24 17:21:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-01-03 00:37:21 +00:00
|
|
|
|
|
|
|
return when(cacheInvalidate);
|
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
|
|
|
|
* @return {Promise(String)} Resolves to header string
|
|
|
|
*/
|
|
|
|
locationHeader = function (req, result) {
|
2014-05-01 23:42:23 +00:00
|
|
|
var apiRoot = config.urlFor('api'),
|
|
|
|
location,
|
|
|
|
post,
|
|
|
|
notification,
|
|
|
|
parsedUrl = req._parsedUrl.pathname.replace(/\/$/, '').split('/'),
|
|
|
|
endpoint = parsedUrl[4];
|
|
|
|
|
|
|
|
if (req.method === 'POST') {
|
|
|
|
if (result.hasOwnProperty('posts')) {
|
|
|
|
post = result.posts[0];
|
|
|
|
location = apiRoot + '/posts/' + post.id + '/?status=' + post.status;
|
|
|
|
} else if (endpoint === 'notifications') {
|
2014-04-28 22:58:18 +02:00
|
|
|
notification = result.notifications;
|
|
|
|
location = apiRoot + '/notifications/' + notification[0].id;
|
2014-05-01 23:42:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return when(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}
|
|
|
|
*/
|
|
|
|
contentDispositionHeader = function () {
|
2014-05-07 01:28:51 -04:00
|
|
|
// replace ':' with '_' for OS that don't support it
|
|
|
|
var now = (new Date()).toJSON().replace(/:/g, '_');
|
|
|
|
return 'Attachment; filename="ghost-' + now + '.json"';
|
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
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ### Format HTTP Errors
|
|
|
|
* Converts the error response from the API into a format which can be returned over HTTP
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @param {Array} error
|
|
|
|
* @return {{errors: Array, statusCode: number}}
|
|
|
|
*/
|
|
|
|
formatHttpErrors = function (error) {
|
|
|
|
var statusCode = 500,
|
|
|
|
errors = [];
|
|
|
|
|
|
|
|
if (!_.isArray(error)) {
|
|
|
|
error = [].concat(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
_.each(error, function (errorItem) {
|
|
|
|
var errorContent = {};
|
|
|
|
|
|
|
|
//TODO: add logic to set the correct status code
|
|
|
|
statusCode = errorItem.code || 500;
|
|
|
|
|
|
|
|
errorContent.message = _.isString(errorItem) ? errorItem :
|
|
|
|
(_.isObject(errorItem) ? errorItem.message : 'Unknown API Error');
|
|
|
|
errorContent.type = errorItem.type || 'InternalServerError';
|
|
|
|
errors.push(errorContent);
|
|
|
|
});
|
|
|
|
|
|
|
|
return {errors: errors, statusCode: statusCode};
|
|
|
|
};
|
2014-05-07 01:28:51 -04: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
|
|
|
|
*/
|
|
|
|
http = function (apiMethod) {
|
2013-06-25 12:43:15 +01:00
|
|
|
return function (req, res) {
|
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:
|
|
|
|
var object = req.body,
|
|
|
|
options = _.extend({}, req.files, req.query, req.params, {
|
|
|
|
context: {
|
|
|
|
user: (req.session && req.session.user) ? req.session.user : 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
|
|
|
|
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
|
|
|
|
if (_.isEmpty(object)) {
|
|
|
|
object = options;
|
|
|
|
options = {};
|
|
|
|
}
|
2014-05-07 01:28:51 -04: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
|
|
|
return apiMethod(object, options)
|
|
|
|
// Handle adding headers
|
|
|
|
.then(function onSuccess(result) {
|
|
|
|
// Add X-Cache-Invalidate header
|
|
|
|
return cacheInvalidationHeader(req, result)
|
|
|
|
.then(function addCacheHeader(header) {
|
|
|
|
if (header) {
|
|
|
|
res.set({'X-Cache-Invalidate': header});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add Location header
|
|
|
|
return locationHeader(req, result);
|
|
|
|
}).then(function addLocationHeader(header) {
|
|
|
|
if (header) {
|
|
|
|
res.set({'Location': header});
|
2014-05-24 08:05:12 +02:00
|
|
|
// The location header indicates that a new object was created.
|
|
|
|
// In this case the status code should be 201 Created
|
|
|
|
res.status(201);
|
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
|
|
|
}
|
|
|
|
|
|
|
|
// Add Content-Disposition Header
|
|
|
|
if (apiMethod === db.exportContent) {
|
|
|
|
res.set({
|
|
|
|
'Content-Disposition': contentDispositionHeader()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// #### Success
|
|
|
|
// Send a properly formatting HTTP response containing the data with correct headers
|
|
|
|
res.json(result || {});
|
|
|
|
});
|
|
|
|
}).catch(function onError(error) {
|
|
|
|
// #### Error
|
|
|
|
var httpErrors = formatHttpErrors(error);
|
|
|
|
// Send a properly formatted HTTP response containing the errors
|
|
|
|
res.json(httpErrors.statusCode, {errors: httpErrors.errors});
|
2014-05-05 15:51:21 +02: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
|
|
|
// Extras
|
|
|
|
init: init,
|
|
|
|
http: http,
|
|
|
|
// API Endpoints
|
|
|
|
db: db,
|
|
|
|
mail: mail,
|
|
|
|
notifications: notifications,
|
2013-12-06 09:51:35 +01:00
|
|
|
posts: posts,
|
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,
|
2014-05-13 14:33:34 +00:00
|
|
|
themes: themes,
|
2014-05-06 00:38:05 +02:00
|
|
|
users: users,
|
|
|
|
slugs: slugs
|
2013-12-06 09:51:35 +01:00
|
|
|
};
|