var db = require('./database'), utils = require('./../public/src/utils'), user = require('./user'), topics = require('./topics'), categories = require('./categories'), favourites = require('./favourites'), threadTools = require('./threadTools'), postTools = require('./postTools'), categories = require('./categories'), feed = require('./feed'), plugins = require('./plugins'), meta = require('./meta'), async = require('async'), nconf = require('nconf'), validator = require('validator'), winston = require('winston'), gravatar = require('gravatar'), S = require('string'); (function(Posts) { var customUserInfo = {}; Posts.create = function(uid, tid, content, callback) { if (uid === null) { return callback(new Error('invalid-user'), null); } topics.isLocked(tid, function(err, locked) { if(err) { return callback(err, null); } else if(locked) { return callback(new Error('topic-locked'), null); } db.incrObjectField('global', 'nextPid', function(err, pid) { if(err) { return callback(err, null); } plugins.fireHook('filter:post.save', content, function(err, newContent) { if(err) { return callback(err, null); } var timestamp = Date.now(), postData = { 'pid': pid, 'uid': uid, 'tid': tid, 'content': newContent, 'timestamp': timestamp, 'reputation': 0, 'editor': '', 'edited': 0, 'deleted': 0 }; db.setObject('post:' + pid, postData); postData.favourited = false; postData.display_moderator_tools = true; postData.relativeTime = new Date(timestamp).toISOString(); topics.addPostToTopic(tid, pid); topics.increasePostCount(tid); topics.updateTimestamp(tid, timestamp); db.incrObjectField('global', 'postCount'); topics.getTopicFields(tid, ['cid', 'pinned'], function(err, topicData) { var cid = topicData.cid; feed.updateTopic(tid); feed.updateRecent(); db.sortedSetAdd('categories:recent_posts:cid:' + cid, timestamp, pid); if(parseInt(topicData.pinned, 10) === 0) { db.sortedSetAdd('categories:' + cid + ':tid', timestamp, tid); } db.setCount('cid:' + cid + ':active_users', function(err, amount) { if (amount > 15) { db.setRemoveRandom('cid:' + cid + ':active_users'); } categories.addActiveUser(cid, uid); }); }); user.onNewPostMade(uid, tid, pid, timestamp); plugins.fireHook('filter:post.get', postData, function(err, newPostData) { if(err) { return callback(err, null); } postData = newPostData; postTools.parse(postData.content, function(err, content) { if(err) { return callback(err, null); } postData.content = content; plugins.fireHook('action:post.save', postData); db.searchIndex('post', content, pid); callback(null, postData); }); }); }); }); }); }; Posts.reply = function(tid, uid, content, callback) { threadTools.privileges(tid, uid, function(err, privileges) { if (content) { content = content.trim(); } if (!content || content.length < meta.config.minimumPostLength) { return callback(new Error('content-too-short')); } else if (!privileges.write) { return callback(new Error('no-privileges')); } Posts.create(uid, tid, content, function(err, postData) { if(err) { return callback(err, null); } else if(!postData) { callback(new Error('reply-error'), null); } Posts.getCidByPid(postData.pid, function(err, cid) { if(err) { return callback(err, null); } db.delete('cid:' + cid + ':read_by_uid'); }); topics.markAsUnreadForAll(tid, function(err) { if(err) { return callback(err, null); } topics.markAsRead(tid, uid); topics.pushUnreadCount(); }); threadTools.notifyFollowers(tid, uid); Posts.addUserInfoToPost(postData, function(err) { if(err) { return callback(err, null); } callback(null, postData); }); }); }); } Posts.getPostsByTid = function(tid, start, end, callback) { db.getListRange('tid:' + tid + ':posts', start, end, function(err, pids) { if(err) { return callback(err); } if(!pids.length) { return callback(null, []); } plugins.fireHook('filter:post.getTopic', pids, function(err, posts) { if(err) { return callback(err); } if(!posts.length) { return callback(null, []); } Posts.getPostsByPids(pids, function(err, posts) { if(err) { return callback(err); } plugins.fireHook('action:post.gotTopic', posts); callback(null, posts); }); }); }); }; Posts.addUserInfoToPost = function(post, callback) { user.getUserFields(post.uid, ['username', 'userslug', 'reputation', 'postcount', 'picture', 'signature', 'banned'], function(err, userData) { if (err) { return callback(err); } postTools.parseSignature(userData.signature, function(err, signature) { if(err) { return callback(err); } post.username = userData.username || 'anonymous'; post.userslug = userData.userslug || ''; post.user_rep = userData.reputation || 0; post.user_postcount = userData.postcount || 0; post.user_banned = parseInt(userData.banned, 10) === 1; post.picture = userData.picture || gravatar.url('', {}, https = nconf.get('https')); if(meta.config.disableSignatures === undefined || parseInt(meta.config.disableSignatures, 10) === 0) { post.signature = signature; } for (var info in customUserInfo) { if (customUserInfo.hasOwnProperty(info)) { post[info] = userData[info] || customUserInfo[info]; } } plugins.fireHook('filter:posts.custom_profile_info', {profile: "", uid: post.uid, pid: post.pid}, function(err, profile_info) { if(err) { return callback(err); } post.additional_profile_info = profile_info.profile; if (post.editor !== '') { user.getUserFields(post.editor, ['username', 'userslug'], function(err, editorData) { if (err) { return callback(err); } post.editorname = editorData.username; post.editorslug = editorData.userslug; callback(); }); } else { callback(); } }); }); }); }; Posts.getPostSummaryByPids = function(pids, stripTags, callback) { var posts = []; function getPostSummary(pid, callback) { async.waterfall([ function(next) { Posts.getPostFields(pid, ['pid', 'tid', 'content', 'uid', 'timestamp', 'deleted'], function(err, postData) { if (parseInt(postData.deleted, 10) === 1) { return callback(null); } else { postData.relativeTime = new Date(parseInt(postData.timestamp || 0, 10)).toISOString(); next(null, postData); } }); }, function(postData, next) { Posts.addUserInfoToPost(postData, function() { next(null, postData); }); }, function(postData, next) { topics.getTopicFields(postData.tid, ['title', 'cid', 'slug', 'deleted'], function(err, topicData) { if (err) { return callback(err); } else if (parseInt(topicData.deleted, 10) === 1) { return callback(null); } categories.getCategoryFields(topicData.cid, ['name', 'icon', 'slug'], function(err, categoryData) { postData.categoryName = categoryData.name; postData.categoryIcon = categoryData.icon; postData.categorySlug = categoryData.slug; postData.title = validator.sanitize(topicData.title).escape(); postData.topicSlug = topicData.slug; next(null, postData); }) }); }, function(postData, next) { if (postData.content) { postTools.parse(postData.content, function(err, content) { if(err) { return next(err); } if(stripTags) { postData.content = S(content).stripTags().s; } else { postData.content = content; } next(null, postData); }); } else { next(null, postData); } } ], function(err, postData) { if (!err) { posts.push(postData); } callback(err); }); } async.eachSeries(pids, getPostSummary, function(err) { if (!err) { callback(null, posts); } else { callback(err, null); } }); }; Posts.getPostData = function(pid, callback) { db.getObject('post:' + pid, function(err, data) { if(err) { return callback(err, null); } plugins.fireHook('filter:post.get', data, function(err, newData) { if(err) { return callback(err, null); } callback(null, newData); }); }); }; Posts.getPostFields = function(pid, fields, callback) { db.getObjectFields('post:' + pid, fields, function(err, data) { if(err) { return callback(err, null); } // TODO: I think the plugins system needs an optional 'parameters' paramter so I don't have to do this: data = data || {}; data.pid = pid; data.fields = fields; plugins.fireHook('filter:post.getFields', data, function(err, data) { if(err) { return callback(err, null); } callback(null, data); }); }); }; Posts.getPostField = function(pid, field, callback) { Posts.getPostFields(pid, [field], function(err, data) { if(err) { return callback(err, null); } callback(null, data[field]); }); }; Posts.setPostField = function(pid, field, value, callback) { db.setObjectField('post:' + pid, field, value, callback); plugins.fireHook('action:post.setField', { 'pid': pid, 'field': field, 'value': value }); }; Posts.setPostFields = function(pid, data, callback) { db.setObject('post:' + pid, data, callback); }; Posts.getPostsByPids = function(pids, callback) { var keys = []; for(var x=0, numPids=pids.length; x