"use strict"; var async = require('async'), nconf = require('nconf'), winston = require('winston'), db = require('../database'), posts = require('../posts'), plugins = require('../plugins'), privileges = require('../privileges'), meta = require('../meta'), topics = require('../topics'), favourites = require('../favourites'), postTools = require('../postTools'), notifications = require('../notifications'), groups = require('../groups'), user = require('../user'), websockets = require('./index'), socketTopics = require('./topics'), events = require('../events'), utils = require('../../public/src/utils'), SocketPosts = {}; SocketPosts.reply = function(socket, data, callback) { if(!data || !data.tid || !data.content) { return callback(new Error('[[error:invalid-data]]')); } data.uid = socket.uid; data.req = websockets.reqFromSocket(socket); topics.reply(data, function(err, postData) { if (err) { return callback(err); } var result = { posts: [postData], privileges: { 'topics:reply': true }, 'reputation:disabled': parseInt(meta.config['reputation:disabled'], 10) === 1, 'downvote:disabled': parseInt(meta.config['downvote:disabled'], 10) === 1, }; callback(null, postData); socket.emit('event:new_post', result); user.updateOnlineUsers(socket.uid); SocketPosts.notifyOnlineUsers(socket.uid, result); if (data.lock) { socketTopics.doTopicAction('lock', 'event:topic_locked', socket, {tids: [postData.topic.tid], cid: postData.topic.cid}); } }); }; SocketPosts.notifyOnlineUsers = function(uid, result) { var cid = result.posts[0].topic.cid; async.waterfall([ function(next) { user.getUidsFromSet('users:online', 0, -1, next); }, function(uids, next) { privileges.categories.filterUids('read', cid, uids, next); }, function(uids, next) { plugins.fireHook('filter:sockets.sendNewPostToUids', {uidsTo: uids, uidFrom: uid, type: 'newPost'}, next); } ], function(err, data) { if (err) { return winston.error(err.stack); } var uids = data.uidsTo; for(var i=0; i parseInt(meta.config.maximumTitleLength, 10)) { return callback(new Error('[[error:title-too-long, ' + meta.config.maximumTitleLength + ']]')); } else if (!data.content || data.content.length < parseInt(meta.config.minimumPostLength, 10)) { return callback(new Error('[[error:content-too-short, ' + meta.config.minimumPostLength + ']]')); } else if (data.content.length > parseInt(meta.config.maximumPostLength, 10)) { return callback(new Error('[[error:content-too-long, ' + meta.config.maximumPostLength + ']]')); } postTools.edit({ uid: socket.uid, handle: data.handle, pid: data.pid, title: data.title, content: data.content, topic_thumb: data.topic_thumb, tags: data.tags }, function(err, result) { if (err) { return callback(err); } if (parseInt(result.post.deleted) !== 1) { websockets.in('topic_' + result.topic.tid).emit('event:post_edited', result); return callback(null, result.post); } socket.emit('event:post_edited', result); callback(null, result.post); async.parallel({ admins: async.apply(groups.getMembers, 'administrators', 0, -1), moderators: async.apply(groups.getMembers, 'cid:' + result.topic.cid + ':privileges:mods', 0, -1), uidsInTopic: async.apply(websockets.getUidsInRoom, 'topic_' + result.topic.tid) }, function(err, results) { if (err) { return winston.error(err); } var uids = results.uidsInTopic.filter(function(uid) { return results.admins.indexOf(uid) !== -1 || results.moderators.indexOf(uid) !== -1; }); uids.forEach(function(uid) { websockets.in('uid_' + uid).emit('event:post_edited', result); }); }); }); }; SocketPosts.delete = function(socket, data, callback) { deleteOrRestore('delete', socket, data, callback); }; SocketPosts.restore = function(socket, data, callback) { deleteOrRestore('restore', socket, data, callback); }; function deleteOrRestore(command, socket, data, callback) { if (!data) { return callback(new Error('[[error:invalid-data]]')); } postTools[command](socket.uid, data.pid, function(err, postData) { if (err) { return callback(err); } var eventName = command === 'delete' ? 'event:post_deleted' : 'event:post_restored'; websockets.in('topic_' + data.tid).emit(eventName, postData); events.log({ type: command === 'delete' ? 'post-delete' : 'post-restore', uid: socket.uid, pid: data.pid, ip: socket.ip }); callback(); }); } SocketPosts.purge = function(socket, data, callback) { function purgePost() { postTools.purge(socket.uid, data.pid, function(err) { if (err) { return callback(err); } websockets.in('topic_' + data.tid).emit('event:post_purged', data.pid); events.log({ type: 'post-purge', uid: socket.uid, pid: data.pid, ip: socket.ip }); callback(); }); } if (!data || !parseInt(data.pid, 10)) { return callback(new Error('[[error:invalid-data]]')); } isMainAndLastPost(data.pid, function(err, results) { if (err) { return callback(err); } if (!results.isMain) { return purgePost(); } if (!results.isLast) { return callback(new Error('[[error:cant-purge-main-post]]')); } posts.getTopicFields(data.pid, ['tid', 'cid'], function(err, topic) { if (err) { return callback(err); } socketTopics.doTopicAction('delete', 'event:topic_deleted', socket, {tids: [topic.tid], cid: topic.cid}, callback); }); }); }; function isMainAndLastPost(pid, callback) { async.parallel({ isMain: function(next) { posts.isMain(pid, next); }, isLast: function(next) { posts.getTopicFields(pid, ['postcount'], function(err, topic) { next(err, topic ? parseInt(topic.postcount, 10) === 1 : false); }); } }, callback); } SocketPosts.getPrivileges = function(socket, pids, callback) { privileges.posts.get(pids, socket.uid, function(err, privileges) { if (err) { return callback(err); } if (!Array.isArray(privileges) || !privileges.length) { return callback(new Error('[[error:invalid-data]]')); } callback(null, privileges); }); }; SocketPosts.getUpvoters = function(socket, pids, callback) { if (!Array.isArray(pids)) { return callback(new Error('[[error:invalid-data]]')); } favourites.getUpvotedUidsByPids(pids, function(err, data) { if (err || !Array.isArray(data) || !data.length) { return callback(err, []); } async.map(data, function(uids, next) { var otherCount = 0; if (uids.length > 6) { otherCount = uids.length - 5; uids = uids.slice(0, 5); } user.getUsernamesByUids(uids, function(err, usernames) { next(err, { otherCount: otherCount, usernames: usernames }); }); }, callback); }); }; SocketPosts.flag = function(socket, pid, callback) { if (!socket.uid) { return callback(new Error('[[error:not-logged-in]]')); } var message = '', userName = '', post; async.waterfall([ function(next) { user.getUserFields(socket.uid, ['username', 'reputation'], next); }, function(userData, next) { if (parseInt(userData.reputation, 10) < parseInt(meta.config['privileges:flag'] || 1, 10)) { return next(new Error('[[error:not-enough-reputation-to-flag]]')); } userName = userData.username; posts.getPostFields(pid, ['pid', 'tid', 'uid', 'content', 'deleted'], next); }, function(postData, next) { if (parseInt(postData.deleted, 10) === 1) { return next(new Error('[[error:post-deleted]]')); } post = postData; posts.flag(post, socket.uid, next); }, function(next) { topics.getTopicFields(post.tid, ['title', 'cid'], next); }, function(topic, next) { post.topic = topic; message = '[[notifications:user_flagged_post_in, ' + userName + ', ' + topic.title + ']]'; posts.parsePost(post, next); }, function(post, next) { async.parallel({ admins: function(next) { groups.getMembers('administrators', 0, -1, next); }, moderators: function(next) { groups.getMembers('cid:' + post.topic.cid + ':privileges:mods', 0, -1, next); } }, next); }, function(results, next) { notifications.create({ bodyShort: message, bodyLong: post.content, pid: pid, nid: 'post_flag:' + pid + ':uid:' + socket.uid, from: socket.uid }, function(err, notification) { if (err || !notification) { return next(err); } notifications.push(notification, results.admins.concat(results.moderators), next); }); } ], callback); }; SocketPosts.loadMoreFavourites = function(socket, data, callback) { loadMorePosts('uid:' + data.uid + ':favourites', socket.uid, data, callback); }; SocketPosts.loadMoreUserPosts = function(socket, data, callback) { loadMorePosts('uid:' + data.uid + ':posts', socket.uid, data, callback); }; function loadMorePosts(set, uid, data, callback) { if (!data || !utils.isNumber(data.uid) || !utils.isNumber(data.after)) { return callback(new Error('[[error:invalid-data]]')); } var start = Math.max(0, parseInt(data.after, 10)), stop = start + 9; posts.getPostSummariesFromSet(set, uid, start, stop, callback); } SocketPosts.getRecentPosts = function(socket, data, callback) { if(!data || !data.count) { return callback(new Error('[[error:invalid-data]]')); } posts.getRecentPosts(socket.uid, 0, data.count - 1, data.term, callback); }; SocketPosts.getCategory = function(socket, pid, callback) { posts.getCidByPid(pid, callback); }; SocketPosts.getPidIndex = function(socket, pid, callback) { posts.getPidIndex(pid, socket.uid, callback); }; module.exports = SocketPosts;