var RDB = require('./redis.js'), utils = require('./utils.js'), marked = require('marked'), user = require('./user.js'), async = require('async'); (function(Posts) { //data structure //*global:next_post_id // *pid:1:content // *pid:1:uid // *pid:1:timestamp // ***pid:1:replies // *uid:1:posts Posts.get = function(callback, tid, current_user, start, end) { if (start == null) start = 0; if (end == null) end = start + 10; async.parallel({ details: function(callback) { RDB.mget([ 'tid:' + tid + ':title', 'tid:' + tid + ':locked' ], function(results) { callback(null, { 'topic_name': results[0], 'locked': results[1] }); }); }, posts: function(callback) { var participant_uids = [], post_calls = []; async.waterfall([ function(next) { RDB.lrange('tid:' + tid + ':posts', start, end, function(pids) { var content = [], uids = [], participants = [], timestamp = [], pid = [], post_rep = []; for (var i=0, ii=pids.length; i<ii; i++) { content.push('pid:' + pids[i] + ':content'); uids.push('pid:' + pids[i] + ':uid'); timestamp.push('pid:' + pids[i] + ':timestamp'); post_rep.push('pid:' + pids[i] + ':rep'); pid.push(pids[i]); } if (pids.length > 0) { RDB.multi() .mget(content) .mget(uids) .mget(timestamp) .mget(post_rep) .exec(function(err, replies) { // Populate uids array for(var x=0,numReplies=replies[1].length;x<numReplies;x++) { var uid = parseInt(replies[1][x]); if (participants.indexOf(uid) === -1) participants.push(uid); } // Construct return object next(null, { replies: replies, pids: pids, participants: participants }); } ); } }); }, function(returnObj, next) { // Get user details var details = {}, calls = []; for(var x=0,numParticipants=returnObj.participants.length;x<numParticipants;x++) { (function(uid) { calls.push(function(next) { // Get individual participant's details user.getUserData(uid, function(userData) { next(null, userData); }) }); })(returnObj.participants[x]); } async.parallel(calls, function(err, results) { for(var x=0,numResults=results.length;x<numResults;x++) { details[returnObj.participants[x]] = results[x]; } returnObj.participants = details; next(null, returnObj); }); }, function(returnObj, next) { // Favourited? var calls = [], numPosts = returnObj.pids.length; for(var x=0;x<numPosts;x++) { (function(pid) { calls.push(function(callback) { Posts.hasFavourited(pid, current_user, function(hasFavourited) { callback(null, hasFavourited); }); }); })(returnObj.pids[x]); } async.parallel(calls, function(err, results) { returnObj.favourites = results; next(null, returnObj); }); } ], function(err, returnObj) { callback(null, returnObj); }); } }, function(err, results) { // Construct posts array var posts = [], participant_details = results.posts.participants; for(var x=0,numPosts=results.posts.pids.length;x<numPosts;x++) { var uid = results.posts.replies[1][x], participant = participant_details[uid]; posts.push({ pid: results.posts.pids[x], content: marked(results.posts.replies[0][x] || ''), uid: uid, timestamp: results.posts.replies[2][x], relativeTime: utils.relativeTime(results.posts.replies[2][x]), post_rep: results.posts.replies[3][x], display_moderator_tools: results.posts.replies[1][x] === current_user ? 'show' : 'hide', username: participant.username, gravatar: participant.picture, user_rep: participant.reputation, fav_star_class: results.posts.favourites[x] ? 'icon-star' : 'icon-star-empty', }); } // Construct return object callback({ 'topic_name': results.details.topic_name, 'locked': parseInt(results.details.locked) || 0, 'topic_id': tid, posts: posts, uids: results.posts.participants }); }); } Posts.reply = function(socket, tid, uid, content) { Posts.create(uid, tid, content, function(pid) { RDB.rpush('tid:' + tid + ':posts', pid); socket.emit('event:alert', { title: 'Reply Successful', message: 'You have successfully replied. Click here to view your reply.', type: 'notify', timeout: 2000 }); user.getUserFields(uid, ['username','reputation','picture'], function(data){ var timestamp = new Date().getTime(); socket.in('topic_' + tid).emit('event:new_post', { 'posts' : [ { 'pid' : pid, 'content' : marked(content || ''), 'uid' : uid, 'username' : data.username || 'anonymous', 'user_rep' : data.reputation || 0, 'post_rep' : 0, 'gravatar' : data.picture, 'timestamp' : timestamp, 'relativeTime': utils.relativeTime(timestamp), 'fav_star_class' :'icon-star-empty' } ] }); }); }); }; Posts.create = function(uid, tid, content, callback) { if (uid === null) return; RDB.incr('global:next_post_id', function(pid) { // Posts Info RDB.set('pid:' + pid + ':content', content); RDB.set('pid:' + pid + ':uid', uid); RDB.set('pid:' + pid + ':timestamp', new Date().getTime()); RDB.set('pid:' + pid + ':rep', 0); RDB.incr('tid:' + tid + ':postcount'); // User Details - move this out later RDB.lpush('uid:' + uid + ':posts', pid); RDB.db.hincrby(uid, 'postcount', 1); if (callback) callback(pid); }); } Posts.favourite = function(io, pid, room_id, uid) { RDB.get('pid:' + pid + ':uid', function(uid_of_poster) { Posts.hasFavourited(pid, uid, function(hasFavourited) { if (hasFavourited == false) { RDB.sadd('pid:' + pid + ':users_favourited', uid); RDB.db.hincrby(String(uid_of_poster), 'reputation', 1); RDB.incr('pid:' + pid + ':rep'); if (room_id) { io.sockets.in(room_id).emit('event:rep_up', {uid: uid_of_poster, pid: pid}); } } }); }); } Posts.unfavourite = function(io, pid, room_id, uid) { RDB.get('pid:' + pid + ':uid', function(uid_of_poster) { Posts.hasFavourited(pid, uid, function(hasFavourited) { if (hasFavourited == true) { RDB.srem('pid:' + pid + ':users_favourited', uid); RDB.db.hincrby(String(uid_of_poster), 'reputation', -1); RDB.decr('pid:' + pid + ':rep'); if (room_id) { io.sockets.in(room_id).emit('event:rep_down', {uid: uid_of_poster, pid: pid}); } } }); }); } Posts.hasFavourited = function(pid, uid, callback) { RDB.sismember('pid:' + pid + ':users_favourited', uid, function(hasFavourited) { callback(hasFavourited); }); } }(exports));