mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-20 22:42:53 -05:00
c02ebb0dcf
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
195 lines
No EOL
6.4 KiB
JavaScript
195 lines
No EOL
6.4 KiB
JavaScript
// # Posts API
|
|
var when = require('when'),
|
|
_ = require('lodash'),
|
|
dataProvider = require('../models'),
|
|
canThis = require('../permissions').canThis,
|
|
errors = require('../errors'),
|
|
utils = require('./utils'),
|
|
|
|
docName = 'posts',
|
|
allowedIncludes = ['created_by', 'updated_by', 'published_by', 'author', 'tags', 'fields'],
|
|
posts;
|
|
|
|
// ## Helpers
|
|
function prepareInclude(include) {
|
|
var index;
|
|
|
|
include = _.intersection(include.split(','), allowedIncludes);
|
|
index = include.indexOf('author');
|
|
|
|
if (index !== -1) {
|
|
include[index] = 'author_id';
|
|
}
|
|
|
|
return include;
|
|
}
|
|
|
|
// ## API Methods
|
|
posts = {
|
|
|
|
/**
|
|
* ### Browse
|
|
* Find a paginated set of posts
|
|
* @param {{context, page, limit, status, staticPages, tag}} options (optional)
|
|
* @returns {Promise(Posts)} Posts Collection with Meta
|
|
*/
|
|
browse: function browse(options) {
|
|
options = options || {};
|
|
|
|
// only published posts if no user is present
|
|
if (!(options.context && options.context.user)) {
|
|
options.status = 'published';
|
|
}
|
|
|
|
if (options.include) {
|
|
options.include = prepareInclude(options.include);
|
|
}
|
|
|
|
return dataProvider.Post.findPage(options);
|
|
},
|
|
|
|
/**
|
|
* ### Read
|
|
* Find a post, by ID or Slug
|
|
* @param {{id_or_slug (required), context, status, include, ...}} options
|
|
* @return {Promise(Post)} Post
|
|
*/
|
|
read: function read(options) {
|
|
var attrs = ['id', 'slug', 'status'],
|
|
data = _.pick(options, attrs);
|
|
options = _.omit(options, attrs);
|
|
|
|
// only published posts if no user is present
|
|
if (!(options.context && options.context.user)) {
|
|
data.status = 'published';
|
|
}
|
|
|
|
if (options.include) {
|
|
options.include = prepareInclude(options.include);
|
|
}
|
|
|
|
return dataProvider.Post.findOne(data, options).then(function (result) {
|
|
if (result) {
|
|
return { posts: [ result.toJSON() ]};
|
|
}
|
|
|
|
return when.reject(new errors.NotFoundError('Post not found.'));
|
|
|
|
});
|
|
},
|
|
|
|
/**
|
|
* ### Edit
|
|
* Update properties of a post
|
|
* @param {Post} object Post or specific properties to update
|
|
* @param {{id (required), context, include,...}} options
|
|
* @return {Promise(Post)} Edited Post
|
|
*/
|
|
edit: function edit(object, options) {
|
|
return canThis(options.context).edit.post(options.id).then(function () {
|
|
return utils.checkObject(object, docName).then(function (checkedPostData) {
|
|
if (options.include) {
|
|
options.include = prepareInclude(options.include);
|
|
}
|
|
|
|
return dataProvider.Post.edit(checkedPostData.posts[0], options);
|
|
}).then(function (result) {
|
|
if (result) {
|
|
var post = result.toJSON();
|
|
|
|
// If previously was not published and now is, signal the change
|
|
if (result.updated('status') !== result.get('status')) {
|
|
post.statusChanged = true;
|
|
}
|
|
return { posts: [ post ]};
|
|
}
|
|
|
|
return when.reject(new errors.NotFoundError('Post not found.'));
|
|
});
|
|
}, function () {
|
|
return when.reject(new errors.NoPermissionError('You do not have permission to edit this post.'));
|
|
});
|
|
},
|
|
|
|
/**
|
|
* ### Add
|
|
* Create a new post along with any tags
|
|
* @param {Post} object
|
|
* @param {{context, include,...}} options
|
|
* @return {Promise(Post)} Created Post
|
|
*/
|
|
add: function add(object, options) {
|
|
options = options || {};
|
|
|
|
return canThis(options.context).create.post().then(function () {
|
|
return utils.checkObject(object, docName).then(function (checkedPostData) {
|
|
if (options.include) {
|
|
options.include = prepareInclude(options.include);
|
|
}
|
|
|
|
return dataProvider.Post.add(checkedPostData.posts[0], options);
|
|
}).then(function (result) {
|
|
var post = result.toJSON();
|
|
|
|
if (post.status === 'published') {
|
|
// When creating a new post that is published right now, signal the change
|
|
post.statusChanged = true;
|
|
}
|
|
return { posts: [ post ]};
|
|
});
|
|
}, function () {
|
|
return when.reject(new errors.NoPermissionError('You do not have permission to add posts.'));
|
|
});
|
|
},
|
|
|
|
|
|
/**
|
|
* ### Destroy
|
|
* Delete a post, cleans up tag relations, but not unused tags
|
|
* @param {{id (required), context,...}} options
|
|
* @return {Promise(Post)} Deleted Post
|
|
*/
|
|
destroy: function destroy(options) {
|
|
return canThis(options.context).remove.post(options.id).then(function () {
|
|
var readOptions = _.extend({}, options, {status: 'all'});
|
|
return posts.read(readOptions).then(function (result) {
|
|
return dataProvider.Post.destroy(options).then(function () {
|
|
var deletedObj = result;
|
|
|
|
if (deletedObj.posts) {
|
|
_.each(deletedObj.posts, function (post) {
|
|
post.statusChanged = true;
|
|
});
|
|
}
|
|
|
|
return deletedObj;
|
|
});
|
|
});
|
|
}, function () {
|
|
return when.reject(new errors.NoPermissionError('You do not have permission to remove posts.'));
|
|
});
|
|
},
|
|
|
|
/**
|
|
* ## Generate Slug
|
|
* Create a unique slug for a given post title
|
|
* @param {{title (required), transacting}} options
|
|
* @returns {Promise(String)} Unique string
|
|
*/
|
|
generateSlug: function generateSlug(options) {
|
|
return canThis(options.context).slug.post().then(function () {
|
|
return dataProvider.Base.Model.generateSlug(dataProvider.Post, options.title, {status: 'all'})
|
|
.then(function (slug) {
|
|
if (slug) {
|
|
return slug;
|
|
}
|
|
return when.reject(new errors.InternalServerError('Could not generate slug'));
|
|
});
|
|
}, function () {
|
|
return when.reject(new errors.NoPermissionError('You do not have permission.'));
|
|
});
|
|
}
|
|
|
|
};
|
|
|
|
module.exports = posts; |