Files
NodeBB/src/topics.js

523 lines
14 KiB
JavaScript
Raw Normal View History

2014-02-26 20:38:49 -05:00
"use strict";
2013-12-01 21:12:05 -05:00
var async = require('async'),
2014-09-06 03:25:32 -04:00
winston = require('winston'),
2013-12-01 21:12:05 -05:00
validator = require('validator'),
2014-08-26 13:47:48 -04:00
_ = require('underscore'),
2013-12-02 21:20:55 -05:00
db = require('./database'),
posts = require('./posts'),
2014-08-07 16:20:46 -04:00
utils = require('../public/src/utils'),
2014-02-20 02:05:49 -05:00
plugins = require('./plugins'),
user = require('./user'),
categories = require('./categories'),
privileges = require('./privileges');
(function(Topics) {
2013-07-02 16:24:13 -04:00
require('./topics/create')(Topics);
2014-06-10 14:24:50 -04:00
require('./topics/delete')(Topics);
2014-03-21 15:40:37 -04:00
require('./topics/unread')(Topics);
2014-03-21 17:04:15 -04:00
require('./topics/recent')(Topics);
require('./topics/popular')(Topics);
2014-03-21 15:47:46 -04:00
require('./topics/fork')(Topics);
2014-03-21 17:04:15 -04:00
require('./topics/posts')(Topics);
2014-04-24 20:05:05 -04:00
require('./topics/follow')(Topics);
require('./topics/tags')(Topics);
2014-03-21 15:40:37 -04:00
2013-07-02 16:24:13 -04:00
Topics.getTopicData = function(tid, callback) {
2014-09-20 19:09:45 -04:00
db.getObject('topic:' + tid, function(err, topic) {
if (err || !topic) {
return callback(err);
}
topic.title = validator.escape(topic.title);
topic.relativeTime = utils.toISOString(topic.timestamp);
callback(null, topic);
});
};
Topics.getTopicsData = function(tids, callback) {
var keys = [];
for (var i=0; i<tids.length; ++i) {
keys.push('topic:' + tids[i]);
}
db.getObjects(keys, function(err, topics) {
if (err) {
return callback(err);
}
for (var i=0; i<tids.length; ++i) {
if(topics[i]) {
topics[i].title = validator.escape(topics[i].title);
topics[i].relativeTime = utils.toISOString(topics[i].timestamp);
}
}
callback(null, topics);
2013-07-02 16:24:13 -04:00
});
2014-02-12 10:03:28 -05:00
};
2013-07-02 16:24:13 -04:00
2013-11-04 15:01:01 -05:00
Topics.getTopicDataWithUser = function(tid, callback) {
Topics.getTopicData(tid, function(err, topic) {
2014-02-25 14:03:47 -05:00
if (err || !topic) {
2014-04-09 22:26:23 -04:00
return callback(err || new Error('[[error:no-topic]]'));
}
user.getUserFields(topic.uid, ['username', 'userslug', 'picture'], function(err, userData) {
2014-05-04 14:33:12 -04:00
if (err) {
return callback(err);
}
2014-04-20 15:07:53 -04:00
topic.user = userData;
callback(null, topic);
2013-07-03 12:14:20 -04:00
});
2013-07-02 16:24:13 -04:00
});
2014-02-12 10:03:28 -05:00
};
2013-07-02 16:24:13 -04:00
2014-02-10 14:15:54 -05:00
Topics.getPageCount = function(tid, uid, callback) {
2014-01-24 20:00:56 -05:00
db.sortedSetCard('tid:' + tid + ':posts', function(err, postCount) {
if(err) {
return callback(err);
}
if(!parseInt(postCount, 10)) {
return callback(null, 1);
}
2014-02-10 14:15:54 -05:00
user.getSettings(uid, function(err, settings) {
if(err) {
return callback(err);
}
2014-01-24 20:00:56 -05:00
2014-02-10 14:15:54 -05:00
callback(null, Math.ceil(parseInt(postCount, 10) / settings.postsPerPage));
});
2014-01-24 20:00:56 -05:00
});
2014-02-12 10:03:28 -05:00
};
2014-01-24 20:00:56 -05:00
2014-02-27 23:45:12 -05:00
Topics.getTidPage = function(tid, uid, callback) {
if(!tid) {
2014-04-09 22:26:23 -04:00
return callback(new Error('[[error:invalid-tid]]'));
2014-02-27 23:45:12 -05:00
}
async.parallel({
index: function(next) {
categories.getTopicIndex(tid, next);
},
settings: function(next) {
user.getSettings(uid, next);
}
}, function(err, results) {
if(err) {
return callback(err);
}
callback(null, Math.ceil((results.index + 1) / results.settings.topicsPerPage));
});
};
Topics.getCategoryData = function(tid, callback) {
Topics.getTopicField(tid, 'cid', function(err, cid) {
2014-01-23 19:01:30 -05:00
if(err) {
callback(err);
}
categories.getCategoryData(cid, callback);
});
2014-02-12 10:03:28 -05:00
};
2014-03-21 17:04:15 -04:00
Topics.getTopics = function(set, uid, tids, callback) {
2014-02-14 13:38:10 -05:00
var returnTopics = {
topics: [],
nextStart: 0
2014-01-30 19:46:25 -05:00
};
2014-08-07 00:06:13 -04:00
if (!Array.isArray(tids) || !tids.length) {
2014-02-14 13:38:10 -05:00
return callback(null, returnTopics);
2014-01-30 19:46:25 -05:00
}
privileges.topics.filter('read', tids, uid, function(err, tids) {
if (err) {
return callback(err);
}
2014-02-26 19:55:28 -05:00
Topics.getTopicsByTids(tids, uid, function(err, topicData) {
2014-01-30 19:46:25 -05:00
if(err) {
return callback(err);
}
if(!topicData || !topicData.length) {
2014-02-14 13:38:10 -05:00
return callback(null, returnTopics);
2014-01-30 19:46:25 -05:00
}
db.sortedSetRevRank(set, topicData[topicData.length - 1].tid, function(err, rank) {
if(err) {
2014-02-26 20:38:49 -05:00
return callback(err);
2014-01-30 19:46:25 -05:00
}
2014-02-14 13:38:10 -05:00
returnTopics.nextStart = parseInt(rank, 10) + 1;
returnTopics.topics = topicData;
callback(null, returnTopics);
2014-01-30 19:46:25 -05:00
});
});
});
2014-02-12 10:03:28 -05:00
};
2014-01-30 19:46:25 -05:00
Topics.getTopicsFromSet = function(uid, set, start, end, callback) {
db.getSortedSetRevRange(set, start, end, function(err, tids) {
if(err) {
return callback(err);
}
2014-03-21 17:04:15 -04:00
Topics.getTopics(set, uid, tids, callback);
});
2014-02-12 10:03:28 -05:00
};
2013-08-23 13:14:36 -04:00
2014-02-26 19:55:28 -05:00
Topics.getTopicsByTids = function(tids, uid, callback) {
if (!Array.isArray(tids) || !tids.length) {
2014-02-12 00:09:02 -05:00
return callback(null, []);
}
2013-08-23 13:14:36 -04:00
Topics.getTopicsData(tids, function(err, topics) {
function mapFilter(array, field) {
return array.map(function(topic) {
return topic && topic[field];
}).filter(function(value, index, array) {
2014-09-20 18:07:46 -04:00
return utils.isNumber(value) && array.indexOf(value) === index;
});
}
if (err) {
return callback(err);
}
var uids = mapFilter(topics, 'uid');
var cids = mapFilter(topics, 'cid');
2014-02-25 14:03:47 -05:00
async.parallel({
users: function(next) {
user.getMultipleUserFields(uids, ['uid', 'username', 'userslug', 'picture'], next);
2014-02-25 14:03:47 -05:00
},
categories: function(next) {
categories.getMultipleCategoryFields(cids, ['cid', 'name', 'slug', 'icon', 'bgColor', 'color', 'disabled'], next);
2014-02-25 14:03:47 -05:00
},
hasRead: function(next) {
Topics.hasReadTopics(tids, uid, next);
2014-02-25 14:03:47 -05:00
},
isAdminOrMod: function(next) {
privileges.categories.isAdminOrMod(cids, uid, next);
},
teasers: function(next) {
2014-08-16 21:33:42 -04:00
Topics.getTeasers(tids, uid, next);
2014-05-21 21:15:11 -04:00
},
tags: function(next) {
Topics.getTopicsTagsObjects(tids, next);
2014-01-16 20:53:32 -05:00
}
}, function(err, results) {
if (err) {
return callback(err);
}
2013-08-23 13:14:36 -04:00
2014-08-26 13:47:48 -04:00
var users = _.object(uids, results.users);
var categories = _.object(cids, results.categories);
var isAdminOrMod = {};
cids.forEach(function(cid, index) {
isAdminOrMod[cid] = results.isAdminOrMod[index];
});
for (var i=0; i<topics.length; ++i) {
if (topics[i]) {
topics[i].category = categories[topics[i].cid];
topics[i].category.disabled = parseInt(topics[i].category.disabled, 10) === 1;
topics[i].user = users[topics[i].uid];
topics[i].teaser = results.teasers[i];
topics[i].tags = results.tags[i];
topics[i].pinned = parseInt(topics[i].pinned, 10) === 1;
topics[i].locked = parseInt(topics[i].locked, 10) === 1;
topics[i].deleted = parseInt(topics[i].deleted, 10) === 1;
topics[i].unread = !(results.hasRead[i] && parseInt(uid, 10) !== 0);
topics[i].unreplied = parseInt(topics[i].postcount, 10) <= 1;
}
}
2014-02-12 00:09:02 -05:00
topics = topics.filter(function(topic) {
return topic && !topic.category.disabled &&
2014-08-06 22:04:31 -04:00
(!topic.deleted || (topic.deleted && isAdminOrMod[topic.cid]) ||
parseInt(topic.uid, 10) === parseInt(uid, 10));
});
2014-09-23 13:33:36 -04:00
plugins.fireHook('filter:topics.get', {topics: topics, uid: uid}, function(err, topicData) {
callback(err, topicData.topics)
});
});
2014-02-12 00:09:02 -05:00
});
2014-02-12 10:03:28 -05:00
};
Topics.getTopicWithPosts = function(tid, set, uid, start, end, reverse, callback) {
Topics.getTopicData(tid, function(err, topicData) {
if (err || !topicData) {
2014-04-09 22:26:23 -04:00
return callback(err || new Error('[[error:no-topic]]'));
2013-11-29 13:09:26 -05:00
}
2013-07-02 16:24:13 -04:00
2014-02-26 16:43:21 -05:00
async.parallel({
posts: function(next) {
posts.getPidsFromSet(set, start, end, reverse, function(err, pids) {
if (err) {
return next(err);
}
2014-08-12 15:08:42 -04:00
pids = topicData.mainPid ? [topicData.mainPid].concat(pids) : pids;
if (!pids.length) {
return next(null, []);
}
posts.getPostsByPids(pids, uid, function(err, posts) {
if (err) {
return next(err);
}
Topics.addPostData(posts, uid, next);
});
});
},
category: function(next) {
2014-02-26 16:43:21 -05:00
Topics.getCategoryData(tid, next);
},
threadTools: function(next) {
2014-03-21 17:04:15 -04:00
plugins.fireHook('filter:topic.thread_tools', [], next);
},
tags: function(next) {
Topics.getTopicTagsObjects(tid, next);
2014-02-26 16:43:21 -05:00
}
}, function(err, results) {
if (err) {
return callback(err);
}
2013-08-23 13:14:36 -04:00
2014-02-26 16:43:21 -05:00
topicData.category = results.category;
topicData.posts = results.posts;
topicData.tags = results.tags;
2014-02-26 16:43:21 -05:00
topicData.thread_tools = results.threadTools;
topicData.pageCount = results.pageCount;
topicData.unreplied = parseInt(topicData.postcount, 10) === 1;
2014-06-10 16:56:55 -04:00
topicData.deleted = parseInt(topicData.deleted, 10) === 1;
topicData.locked = parseInt(topicData.locked, 10) === 1;
topicData.pinned = parseInt(topicData.pinned, 10) === 1;
2014-02-26 16:43:21 -05:00
2014-08-06 21:30:41 -04:00
plugins.fireHook('filter:topic.get', topicData, callback);
});
});
2014-02-12 10:03:28 -05:00
};
Topics.getMainPost = function(tid, uid, callback) {
2014-08-18 16:18:51 -04:00
Topics.getMainPosts([tid], uid, function(err, mainPosts) {
callback(err, Array.isArray(mainPosts) && mainPosts.length ? mainPosts[0] : null);
});
};
2014-08-18 16:03:25 -04:00
Topics.getMainPosts = function(tids, uid, callback) {
var keys = tids.map(function(tid) {
return 'topic:' + tid;
});
db.getObjectsFields(keys, ['mainPid'], function(err, topicData) {
if (err) {
return callback(err);
}
2014-08-18 16:03:25 -04:00
var mainPids = topicData.map(function(topic) {
return topic ? topic.mainPid : null;
});
posts.getPostsByPids(mainPids, uid, function(err, postData) {
if (err) {
return callback(err);
}
2014-08-18 16:03:25 -04:00
if (!Array.isArray(postData) || !postData.length) {
return callback(null, []);
}
2014-08-18 16:03:25 -04:00
Topics.addPostData(postData, uid, callback);
});
});
2014-09-06 03:35:28 -04:00
};
2014-02-12 10:03:28 -05:00
2014-08-16 21:33:42 -04:00
Topics.getTeasers = function(tids, uid, callback) {
2014-02-12 10:03:28 -05:00
if(!Array.isArray(tids)) {
return callback(null, []);
}
async.map(tids, function(tid, next) {
db.getSortedSetRevRange('tid:' + tid + ':posts', 0, 0, function(err, data) {
next(err, Array.isArray(data) && data.length ? data[0] : null);
});
}, function(err, pids) {
if (err) {
return callback(err);
}
2014-09-06 00:58:03 -04:00
var postKeys = pids.filter(Boolean).map(function(pid) {
return 'post:' + pid;
});
2014-08-16 21:33:42 -04:00
db.getObjectsFields(postKeys, ['pid', 'uid', 'timestamp', 'tid'], function(err, postData) {
if (err) {
return callback(err);
}
2014-08-16 21:33:42 -04:00
var uids = postData.map(function(post) {
return post.uid;
}).filter(function(uid, index, array) {
return array.indexOf(uid) === index;
});
2014-08-16 21:33:42 -04:00
async.parallel({
users: function(next) {
user.getMultipleUserFields(uids, ['uid', 'username', 'userslug', 'picture'], next);
},
indices: function(next) {
posts.getPostIndices(postData, uid, next);
}
}, function(err, results) {
if (err) {
return callback(err);
}
var users = {};
2014-08-16 21:33:42 -04:00
results.users.forEach(function(user) {
users[user.uid] = user;
});
2014-09-06 00:58:03 -04:00
var tidToPost = {};
2014-08-16 21:33:42 -04:00
postData.forEach(function(post, index) {
post.user = users[post.uid];
2014-08-16 21:33:42 -04:00
post.index = results.indices[index] + 1;
post.timestamp = utils.toISOString(post.timestamp);
2014-09-06 00:58:03 -04:00
tidToPost[post.tid] = post;
});
var teasers = tids.map(function(tid) {
return tidToPost[tid];
});
2014-09-06 00:58:03 -04:00
callback(null, teasers);
});
});
});
2014-02-12 10:03:28 -05:00
};
2014-08-17 00:14:45 -04:00
Topics.getTeaser = function(tid, uid, callback) {
2014-04-24 20:05:05 -04:00
Topics.getLatestUndeletedPid(tid, function(err, pid) {
if (err || !pid) {
2014-02-12 10:03:28 -05:00
return callback(err);
2013-11-15 14:57:50 -05:00
}
async.parallel({
postData: function(next) {
posts.getPostFields(pid, ['pid', 'uid', 'timestamp'], function(err, postData) {
if (err) {
return next(err);
} else if(!postData || !utils.isNumber(postData.uid)) {
2014-06-28 16:18:29 -04:00
return callback();
}
user.getUserFields(postData.uid, ['username', 'userslug', 'picture'], function(err, userData) {
if (err) {
return next(err);
}
postData.user = userData;
next(null, postData);
});
});
},
postIndex: function(next) {
2014-08-17 00:14:45 -04:00
posts.getPidIndex(pid, uid, next);
}
}, function(err, results) {
2013-11-15 14:57:50 -05:00
if (err) {
2014-02-12 10:03:28 -05:00
return callback(err);
2013-11-15 14:57:50 -05:00
}
results.postData.timestamp = utils.toISOString(results.postData.timestamp);
2014-06-13 15:33:22 -04:00
results.postData.index = results.postIndex;
callback(null, results.postData);
2013-11-15 14:57:50 -05:00
});
});
2014-02-26 20:38:49 -05:00
};
2013-07-02 16:24:13 -04:00
Topics.getTopicField = function(tid, field, callback) {
2013-12-02 21:20:55 -05:00
db.getObjectField('topic:' + tid, field, callback);
2014-02-26 20:38:49 -05:00
};
2013-08-23 13:14:36 -04:00
Topics.getTopicFields = function(tid, fields, callback) {
2013-12-02 21:20:55 -05:00
db.getObjectFields('topic:' + tid, fields, callback);
2014-02-26 20:38:49 -05:00
};
2013-07-02 16:24:13 -04:00
Topics.getTopicsFields = function(tids, fields, callback) {
var keys = tids.map(function(tid) {
return 'topic:' + tid;
});
db.getObjectsFields(keys, fields, callback);
};
2013-11-27 15:02:09 -05:00
Topics.setTopicField = function(tid, field, value, callback) {
2013-12-02 21:20:55 -05:00
db.setObjectField('topic:' + tid, field, value, callback);
2014-02-26 20:38:49 -05:00
};
2013-07-02 16:24:13 -04:00
Topics.isLocked = function(tid, callback) {
Topics.getTopicField(tid, 'locked', function(err, locked) {
2013-11-15 14:57:50 -05:00
if(err) {
2014-02-25 14:41:14 -05:00
return callback(err);
2013-11-15 14:57:50 -05:00
}
2013-12-05 13:11:27 -05:00
callback(null, parseInt(locked, 10) === 1);
2013-07-02 16:24:13 -04:00
});
2014-02-26 20:38:49 -05:00
};
2013-07-02 16:24:13 -04:00
2014-06-23 17:26:02 -04:00
Topics.isOwner = function(tid, uid, callback) {
uid = parseInt(uid, 10);
2014-09-20 19:09:45 -04:00
if (!uid) {
return callback(null, false);
}
2014-06-23 17:26:02 -04:00
Topics.getTopicField(tid, 'uid', function(err, author) {
callback(err, parseInt(author, 10) === uid);
2014-06-23 17:26:02 -04:00
});
};
2013-08-27 13:32:43 -04:00
Topics.getUids = function(tid, callback) {
Topics.getPids(tid, function(err, pids) {
2014-03-23 14:25:16 -04:00
if (err) {
return callback(err);
2013-08-27 13:32:43 -04:00
}
2014-03-23 14:25:16 -04:00
var keys = pids.map(function(pid) {
return 'post:' + pid;
});
db.getObjectsFields(keys, ['uid'], function(err, data) {
2014-02-25 14:41:14 -05:00
if (err) {
return callback(err);
}
2013-08-27 13:32:43 -04:00
2014-03-23 14:25:16 -04:00
var uids = data.map(function(data) {
return data.uid;
}).filter(function(uid, pos, array) {
return array.indexOf(uid) === pos;
});
callback(null, uids);
2013-08-27 13:32:43 -04:00
});
});
2014-02-26 20:38:49 -05:00
};
2013-08-27 13:32:43 -04:00
2014-07-24 17:30:37 -04:00
Topics.search = function(tid, term, callback) {
if (plugins.hasListeners('filter:topic.search')) {
plugins.fireHook('filter:topic.search', {
tid: tid,
term: term
}, callback);
} else {
callback(new Error('no-plugins-available'), []);
2014-07-24 17:30:37 -04:00
}
};
2014-04-10 20:31:57 +01:00
}(exports));