From 6485108dfd3e0955421f742f0dc6d33ed5f3c04e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Mon, 10 Jul 2017 00:28:15 -0400 Subject: [PATCH] faster getReplies --- src/topics/posts.js | 124 +++++++++++++++++++++++--------------------- src/user.js | 22 ++++++-- 2 files changed, 83 insertions(+), 63 deletions(-) diff --git a/src/topics/posts.js b/src/topics/posts.js index 8377215609..c09e119cfc 100644 --- a/src/topics/posts.js +++ b/src/topics/posts.js @@ -169,11 +169,9 @@ module.exports = function (Topics) { async.apply(posts.getPostsFields, parentPids, ['uid']), function (_parentPosts, next) { parentPosts = _parentPosts; - var parentUids = parentPosts.map(function (postObj) { - return parseInt(postObj.uid, 10); - }).filter(function (uid, idx, users) { - return users.indexOf(uid) === idx; - }); + var parentUids = _.uniq(parentPosts.map(function (postObj) { + return postObj && parseInt(postObj.uid, 10); + })); user.getUsersFields(parentUids, ['username'], next); }, @@ -391,59 +389,69 @@ module.exports = function (Topics) { }; function getPostReplies(pids, callerUid, callback) { - async.map(pids, function (pid, next) { - var replyPids; - var uids = []; - async.waterfall([ - function (next) { - db.getSortedSetRange('pid:' + pid + ':replies', 0, -1, next); - }, - function (_replyPids, next) { - replyPids = _replyPids; - - var count = 0; - - async.until(function () { - return count === replyPids.length || uids.length === 6; - }, function (next) { - async.waterfall([ - function (next) { - posts.getPostField(replyPids[count], 'uid', next); - }, - function (uid, next) { - uid = parseInt(uid, 10); - if (uids.indexOf(uid) === -1) { - uids.push(uid); - } - count += 1; - next(); - }, - ], next); - }, next); - }, - function (next) { - async.parallel({ - users: function (next) { - user.getUsersWithFields(uids, ['uid', 'username', 'userslug', 'picture'], callerUid, next); - }, - timestampISO: function (next) { - posts.getPostField(replyPids[0], 'timestamp', function (err, timestamp) { - next(err, utils.toISOString(timestamp)); - }); - }, - }, next); - }, - function (replies, next) { - if (replies.users.length > 5) { - replies.users.shift(); - replies.hasMore = true; - } + var arrayOfReplyPids; + var replyData; + var uniqueUids; + var uniquePids; + async.waterfall([ + function (next) { + var keys = pids.map(function (pid) { + return 'pid:' + pid + ':replies'; + }); + db.getSortedSetsMembers(keys, next); + }, + function (arrayOfPids, next) { + arrayOfReplyPids = arrayOfPids; - replies.count = replyPids.length; - replies.text = replies.count > 1 ? '[[topic:replies_to_this_post, ' + replies.count + ']]' : '[[topic:one_reply_to_this_post]]'; - next(null, replies); - }, - ], next); - }, callback); + uniquePids = _.uniq(_.flatten(arrayOfPids)); + + posts.getPostsFields(uniquePids, ['pid', 'uid', 'timestamp'], next); + }, + function (_replyData, next) { + replyData = _replyData; + var uids = replyData.map(function (replyData) { + return replyData && replyData.uid; + }); + + uniqueUids = _.uniq(uids); + + user.getUsersWithFields(uniqueUids, ['uid', 'username', 'userslug', 'picture'], callerUid, next); + }, + function (userData, next) { + var uidMap = _.zipObject(uniqueUids, userData); + var pidMap = _.zipObject(uniquePids, replyData); + + var returnData = arrayOfReplyPids.map(function (replyPids) { + var uidsUsed = {}; + var currentData = { + hasMore: false, + users: [], + text: replyPids.length > 1 ? '[[topic:replies_to_this_post, ' + replyPids.length + ']]' : '[[topic:one_reply_to_this_post]]', + count: replyPids.length, + timestampISO: replyPids.length ? utils.toISOString(pidMap[replyPids[0]].timestamp) : undefined, + }; + replyPids.sort(function(a, b) { + return parseInt(a, 10) - parseInt(b, 10); + }); + + replyPids.forEach(function (replyPid) { + var replyData = pidMap[replyPid]; + if (!uidsUsed[replyData.uid] && currentData.users.length < 6) { + currentData.users.push(uidMap[replyData.uid]); + uidsUsed[replyData.uid] = true; + } + }); + + if (currentData.users.length > 5) { + currentData.users.pop(); + currentData.hasMore = true; + } + + return currentData; + }); + + next(null, returnData); + }, + ], callback); } }; diff --git a/src/user.js b/src/user.js index 9a8251d6ee..db265f76bc 100644 --- a/src/user.js +++ b/src/user.js @@ -78,12 +78,24 @@ User.getUsersWithFields = function (uids, fields, uid, callback) { function (results, next) { results.userData.forEach(function (user, index) { if (user) { - user.status = User.getStatus(user); user.administrator = results.isAdmin[index]; - user.banned = parseInt(user.banned, 10) === 1; - user.banned_until = parseInt(user['banned:expire'], 10) || 0; - user.banned_until_readable = user.banned_until ? new Date(user.banned_until).toString() : 'Not Banned'; - user['email:confirmed'] = parseInt(user['email:confirmed'], 10) === 1; + + if (user.hasOwnProperty('status')) { + user.status = User.getStatus(user); + } + + if (user.hasOwnProperty('banned')) { + user.banned = parseInt(user.banned, 10) === 1; + } + + if (user.hasOwnProperty('banned:expire')) { + user.banned_until = parseInt(user['banned:expire'], 10) || 0; + user.banned_until_readable = user.banned_until ? new Date(user.banned_until).toString() : 'Not Banned'; + } + + if (user.hasOwnProperty(['email:confirmed'])) { + user['email:confirmed'] = parseInt(user['email:confirmed'], 10) === 1; + } } }); plugins.fireHook('filter:userlist.get', { users: results.userData, uid: uid }, next);