Files
NodeBB/src/search.js

374 lines
10 KiB
JavaScript
Raw Normal View History

2014-07-07 17:36:10 -04:00
'use strict';
2016-02-13 09:35:48 +02:00
var async = require('async');
2017-06-25 20:00:05 -04:00
var _ = require('lodash');
2016-02-13 09:35:48 +02:00
var db = require('./database');
var posts = require('./posts');
var topics = require('./topics');
var categories = require('./categories');
var user = require('./user');
var plugins = require('./plugins');
var privileges = require('./privileges');
var utils = require('./utils');
2014-07-07 17:36:10 -04:00
2017-05-26 22:28:12 -04:00
var search = module.exports;
2014-07-07 17:36:10 -04:00
search.search = function (data, callback) {
2014-07-07 17:36:10 -04:00
var start = process.hrtime();
2018-10-18 12:50:24 -04:00
data.searchIn = data.searchIn || 'titlesposts';
2018-11-11 17:57:17 -05:00
data.sortBy = data.sortBy || 'relevance';
2016-02-13 09:35:48 +02:00
async.waterfall([
function (next) {
2018-10-18 12:50:24 -04:00
if (data.searchIn === 'posts' || data.searchIn === 'titles' || data.searchIn === 'titlesposts') {
2016-02-13 09:35:48 +02:00
searchInContent(data, next);
2018-10-18 12:50:24 -04:00
} else if (data.searchIn === 'users') {
2016-02-13 09:35:48 +02:00
user.search(data, next);
2018-10-18 12:50:24 -04:00
} else if (data.searchIn === 'tags') {
2016-02-13 09:35:48 +02:00
topics.searchAndLoadTags(data, next);
} else {
next(new Error('[[error:unknown-search-filter]]'));
}
},
function (result, next) {
result.time = (process.elapsedTimeSince(start) / 1000).toFixed(2);
next(null, result);
2017-02-17 19:31:21 -07:00
},
2016-02-13 09:35:48 +02:00
], callback);
2015-01-07 16:10:11 -05:00
};
2015-03-14 23:00:24 -04:00
function searchInContent(data, callback) {
2015-02-05 14:55:36 -05:00
data.uid = data.uid || 0;
2017-05-26 22:28:12 -04:00
var pids;
var metadata;
2018-11-09 16:48:34 -05:00
var itemsPerPage = Math.min(data.itemsPerPage || 10, 100);
2018-11-11 17:57:17 -05:00
const returnData = {
posts: [],
matchCount: 0,
pageCount: 1,
};
2017-05-26 22:28:12 -04:00
async.waterfall([
function (next) {
async.parallel({
2018-11-10 07:35:46 -05:00
searchCids: async.apply(getSearchCids, data),
searchUids: async.apply(getSearchUids, data),
2017-05-26 22:28:12 -04:00
}, next);
2017-02-17 19:31:21 -07:00
},
2017-05-26 22:28:12 -04:00
function (results, next) {
function doSearch(type, searchIn, next) {
2018-03-27 20:32:57 -04:00
if (searchIn.includes(data.searchIn)) {
plugins.fireHook('filter:search.query', {
index: type,
content: data.query,
matchWords: data.matchWords || 'all',
cid: results.searchCids,
uid: results.searchUids,
2018-06-05 13:43:34 -04:00
searchData: data,
2018-03-27 20:32:57 -04:00
}, next);
2015-03-14 23:00:24 -04:00
} else {
next(null, []);
}
}
2017-05-26 22:28:12 -04:00
async.parallel({
2018-11-10 07:35:46 -05:00
pids: async.apply(doSearch, 'post', ['posts', 'titlesposts']),
tids: async.apply(doSearch, 'topic', ['titles', 'titlesposts']),
2017-05-26 22:28:12 -04:00
}, next);
},
function (results, next) {
pids = results.pids;
2018-11-11 17:57:17 -05:00
if (data.returnIds) {
return callback(null, results);
}
2018-10-18 12:50:24 -04:00
if (!results.pids.length && !results.tids.length) {
2018-11-11 17:57:17 -05:00
return callback(null, returnData);
2015-01-07 16:10:11 -05:00
}
2015-03-14 23:00:24 -04:00
2017-05-26 22:28:12 -04:00
topics.getMainPids(results.tids, next);
},
function (mainPids, next) {
2018-10-21 19:33:46 -04:00
pids = mainPids.concat(pids).filter(Boolean);
2015-03-14 23:00:24 -04:00
privileges.posts.filter('topics:read', pids, data.uid, next);
2017-05-26 22:28:12 -04:00
},
function (pids, next) {
filterAndSort(pids, data, next);
},
function (pids, next) {
plugins.fireHook('filter:search.inContent', {
pids: pids,
}, next);
},
function (_metadata, next) {
metadata = _metadata;
2018-11-11 17:57:17 -05:00
returnData.matchCount = metadata.pids.length;
returnData.pageCount = Math.max(1, Math.ceil(parseInt(returnData.matchCount, 10) / itemsPerPage));
2017-05-26 22:28:12 -04:00
if (data.page) {
2018-11-11 17:57:17 -05:00
const start = Math.max(0, (data.page - 1)) * itemsPerPage;
2018-02-02 11:55:55 -05:00
metadata.pids = metadata.pids.slice(start, start + itemsPerPage);
2017-05-26 22:28:12 -04:00
}
posts.getPostSummaryByPids(metadata.pids, data.uid, {}, next);
2017-05-26 22:28:12 -04:00
},
function (posts, next) {
2018-11-11 17:57:17 -05:00
returnData.posts = posts;
// Append metadata to returned payload (without pids)
delete metadata.pids;
2018-11-11 17:57:17 -05:00
next(null, Object.assign(returnData, metadata));
2017-05-26 22:28:12 -04:00
},
], callback);
2015-01-07 16:10:11 -05:00
}
2014-07-07 17:36:10 -04:00
2015-03-14 23:00:24 -04:00
function filterAndSort(pids, data, callback) {
2018-11-11 17:57:17 -05:00
if (data.sortBy === 'relevance' && !data.replies && !data.timeRange && !data.hasTags) {
return setImmediate(callback, null, pids);
}
2017-05-26 22:28:12 -04:00
async.waterfall([
function (next) {
getMatchedPosts(pids, data, next);
},
function (posts, next) {
2017-05-27 00:30:07 -04:00
if (!posts.length) {
2017-05-26 22:28:12 -04:00
return callback(null, pids);
}
posts = posts.filter(Boolean);
2017-05-26 22:28:12 -04:00
posts = filterByPostcount(posts, data.replies, data.repliesFilter);
posts = filterByTimerange(posts, data.timeRange, data.timeFilter);
posts = filterByTags(posts, data.hasTags);
2017-05-26 22:28:12 -04:00
sortPosts(posts, data);
2017-06-07 14:21:03 -04:00
plugins.fireHook('filter:search.filterAndSort', { pids: pids, posts: posts, data: data }, next);
},
function (result, next) {
2018-10-21 19:33:46 -04:00
pids = result.posts.map(post => post && post.pid);
2017-05-26 22:28:12 -04:00
next(null, pids);
},
], callback);
}
2015-03-14 23:00:24 -04:00
function getMatchedPosts(pids, data, callback) {
2018-11-10 07:35:46 -05:00
var postFields = ['pid', 'uid', 'tid', 'timestamp', 'deleted', 'upvotes', 'downvotes'];
var categoryFields = [];
2018-11-11 17:57:17 -05:00
if (data.sortBy.startsWith('category.')) {
2018-11-10 07:35:46 -05:00
categoryFields.push(data.sortBy.split('.')[1]);
}
2018-10-21 19:33:46 -04:00
var postsData;
2018-11-10 07:35:46 -05:00
let tids;
let uids;
async.waterfall([
function (next) {
2018-10-21 19:33:46 -04:00
posts.getPostsFields(pids, postFields, next);
},
2018-10-21 19:33:46 -04:00
function (_postsData, next) {
postsData = _postsData.filter(post => post && !post.deleted);
async.parallel({
users: function (next) {
2018-11-11 17:57:17 -05:00
if (data.sortBy.startsWith('user')) {
2018-11-10 07:35:46 -05:00
uids = _.uniq(postsData.map(post => post.uid));
2015-09-25 17:38:58 -04:00
user.getUsersFields(uids, ['username'], next);
} else {
next();
}
},
topics: function (next) {
2016-12-19 21:40:09 +03:00
var topicsData;
2018-11-10 07:35:46 -05:00
tids = _.uniq(postsData.map(post => post.tid));
let cids;
async.waterfall([
function (next) {
2018-10-21 19:33:46 -04:00
topics.getTopicsData(tids, next);
},
function (_topics, next) {
2016-12-19 21:40:09 +03:00
topicsData = _topics;
async.parallel({
categories: function (next) {
if (!categoryFields.length) {
return next();
}
2018-11-10 07:35:46 -05:00
cids = _.uniq(topicsData.map(topic => topic && topic.cid));
db.getObjectsFields(cids.map(cid => 'category:' + cid), categoryFields, next);
2016-12-19 21:40:09 +03:00
},
tags: function (next) {
2017-12-11 11:21:22 -05:00
if (Array.isArray(data.hasTags) && data.hasTags.length) {
2016-12-19 21:40:09 +03:00
topics.getTopicsTags(tids, next);
} else {
setImmediate(next);
}
2017-02-17 19:31:21 -07:00
},
}, next);
2017-02-17 19:31:21 -07:00
},
2017-05-26 22:28:12 -04:00
function (results, next) {
2018-11-10 07:35:46 -05:00
const cidToCategory = _.zipObject(cids, results.categories);
2017-05-26 22:28:12 -04:00
topicsData.forEach(function (topic, index) {
2018-11-10 07:35:46 -05:00
if (topic && results.categories && cidToCategory[topic.cid]) {
topic.category = cidToCategory[topic.cid];
2017-05-26 22:28:12 -04:00
}
if (topic && results.tags && results.tags[index]) {
topic.tags = results.tags[index];
}
});
2017-05-26 22:28:12 -04:00
next(null, topicsData);
},
], next);
2017-02-17 19:31:21 -07:00
},
}, next);
},
function (results, next) {
2018-11-10 07:35:46 -05:00
const tidToTopic = _.zipObject(tids, results.topics);
const uidToUser = _.zipObject(uids, results.users);
postsData.forEach(function (post) {
if (results.topics && tidToTopic[post.tid]) {
post.topic = tidToTopic[post.tid];
if (post.topic && post.topic.category) {
post.category = post.topic.category;
}
}
2018-11-10 07:35:46 -05:00
if (uidToUser[post.uid]) {
post.user = uidToUser[post.uid];
}
});
2018-10-21 19:33:46 -04:00
postsData = postsData.filter(post => post && post.topic && !post.topic.deleted);
next(null, postsData);
2017-02-17 19:31:21 -07:00
},
], callback);
}
function filterByPostcount(posts, postCount, repliesFilter) {
postCount = parseInt(postCount, 10);
if (postCount) {
if (repliesFilter === 'atleast') {
2018-11-10 07:35:46 -05:00
posts = posts.filter(post => post.topic && post.topic.postcount >= postCount);
} else {
2018-11-10 07:35:46 -05:00
posts = posts.filter(post => post.topic && post.topic.postcount <= postCount);
}
}
return posts;
}
function filterByTimerange(posts, timeRange, timeFilter) {
2017-02-18 14:29:52 -07:00
timeRange = parseInt(timeRange, 10) * 1000;
if (timeRange) {
2018-11-10 07:35:46 -05:00
const time = Date.now() - timeRange;
if (timeFilter === 'newer') {
2018-11-10 07:35:46 -05:00
posts = posts.filter(post => post.timestamp >= time);
} else {
2018-11-10 07:35:46 -05:00
posts = posts.filter(post => post.timestamp <= time);
}
}
return posts;
}
2016-12-19 21:40:09 +03:00
function filterByTags(posts, hasTags) {
2017-12-11 11:21:22 -05:00
if (Array.isArray(hasTags) && hasTags.length) {
2016-12-19 21:40:09 +03:00
posts = posts.filter(function (post) {
var hasAllTags = false;
2017-12-11 11:21:22 -05:00
if (post && post.topic && Array.isArray(post.topic.tags) && post.topic.tags.length) {
2018-11-10 07:35:46 -05:00
hasAllTags = hasTags.every(tag => post.topic.tags.includes(tag));
2016-12-19 21:40:09 +03:00
}
return hasAllTags;
});
}
return posts;
}
function sortPosts(posts, data) {
2018-11-11 17:57:17 -05:00
if (!posts.length || data.sortBy === 'relevance') {
return;
}
data.sortDirection = data.sortDirection || 'desc';
2015-10-11 21:56:28 -04:00
var direction = data.sortDirection === 'desc' ? 1 : -1;
2018-11-10 07:35:46 -05:00
const fields = data.sortBy.split('.');
if (fields.length === 1) {
return posts.sort((p1, p2) => direction * (p2[fields[0]] - p1[fields[0]]));
}
var firstPost = posts[0];
if (!fields || fields.length !== 2 || !firstPost[fields[0]] || !firstPost[fields[0]][fields[1]]) {
return;
}
2015-10-11 21:56:28 -04:00
var isNumeric = utils.isNumber(firstPost[fields[0]][fields[1]]);
if (isNumeric) {
2018-11-10 07:35:46 -05:00
posts.sort((p1, p2) => direction * (p2[fields[0]][fields[1]] - p1[fields[0]][fields[1]]));
} else {
posts.sort(function (p1, p2) {
2015-09-27 14:56:27 -04:00
if (p1[fields[0]][fields[1]] > p2[fields[0]][fields[1]]) {
return direction;
} else if (p1[fields[0]][fields[1]] < p2[fields[0]][fields[1]]) {
return -direction;
}
return 0;
});
}
}
2015-03-14 23:00:24 -04:00
function getSearchCids(data, callback) {
if (!Array.isArray(data.categories) || !data.categories.length) {
2015-02-05 14:55:36 -05:00
return callback(null, []);
}
if (data.categories.includes('all')) {
return categories.getCidsByPrivilege('categories:cid', data.uid, 'read', callback);
}
2017-05-26 22:28:12 -04:00
async.waterfall([
function (next) {
async.parallel({
watchedCids: function (next) {
if (data.categories.includes('watched')) {
user.getCategoriesByStates(data.uid, [categories.watchStates.watching], next);
2017-05-26 22:28:12 -04:00
} else {
2018-11-11 17:57:17 -05:00
setImmediate(next, null, []);
2017-05-26 22:28:12 -04:00
}
},
childrenCids: function (next) {
if (data.searchChildren) {
getChildrenCids(data.categories, data.uid, next);
} else {
2018-11-11 17:57:17 -05:00
setImmediate(next, null, []);
2017-05-26 22:28:12 -04:00
}
},
}, next);
2015-02-05 14:55:36 -05:00
},
2017-05-26 22:28:12 -04:00
function (results, next) {
2018-11-10 07:35:46 -05:00
const cids = _.uniq(results.watchedCids.concat(results.childrenCids).concat(data.categories).filter(Boolean));
2017-05-26 22:28:12 -04:00
next(null, cids);
2017-02-17 19:31:21 -07:00
},
2017-05-26 22:28:12 -04:00
], callback);
2015-02-05 14:55:36 -05:00
}
function getChildrenCids(cids, uid, callback) {
2017-05-26 22:28:12 -04:00
async.waterfall([
function (next) {
2018-11-10 07:35:46 -05:00
async.map(cids, categories.getChildrenCids, next);
2017-05-26 22:28:12 -04:00
},
2018-11-10 07:35:46 -05:00
function (childrenCids, next) {
privileges.categories.filterCids('find', _.uniq(_.flatten(childrenCids)), uid, next);
2017-05-26 22:28:12 -04:00
},
], callback);
2015-02-05 14:55:36 -05:00
}
2015-03-14 23:00:24 -04:00
function getSearchUids(data, callback) {
if (data.postedBy) {
2015-03-15 01:45:24 -04:00
user.getUidsByUsernames(Array.isArray(data.postedBy) ? data.postedBy : [data.postedBy], callback);
2015-03-14 23:00:24 -04:00
} else {
2018-11-11 17:57:17 -05:00
setImmediate(callback, null, []);
2015-03-14 23:00:24 -04:00
}
}
Async refactor in place (#7736) * feat: allow both callback&and await * feat: ignore async key * feat: callbackify and promisify in same file * Revert "feat: callbackify and promisify in same file" This reverts commit cea206a9b8e6d8295310074b18cc82a504487862. * feat: no need to store .callbackify * feat: change getTopics to async * feat: remove .async * fix: byScore * feat: rewrite topics/index and social with async/await * fix: rewrite topics/data.js fix issue with async.waterfall, only pass result if its not undefined * feat: add callbackify to redis/psql * feat: psql use await * fix: redis :volcano: * feat: less returns * feat: more await rewrite * fix: redis tests * feat: convert sortedSetAdd rewrite psql transaction to async/await * feat: :dog: * feat: test * feat: log client and query * feat: log bind * feat: more logs * feat: more logs * feat: check perform * feat: dont callbackify transaction * feat: remove logs * fix: main functions * feat: more logs * fix: increment * fix: rename * feat: remove cls * fix: remove console.log * feat: add deprecation message to .async usage * feat: update more dbal methods * fix: redis :voodoo: * feat: fix redis zrem, convert setObject * feat: upgrade getObject methods * fix: psql getObjectField * fix: redis tests * feat: getObjectKeys * feat: getObjectValues * feat: isObjectField * fix: add missing return * feat: delObjectField * feat: incrObjectField * fix: add missing await * feat: remove exposed helpers * feat: list methods * feat: flush/empty * feat: delete * fix: redis delete all * feat: get/set * feat: incr/rename * feat: type * feat: expire * feat: setAdd * feat: setRemove * feat: isSetMember * feat: getSetMembers * feat: setCount, setRemoveRandom * feat: zcard,zcount * feat: sortedSetRank * feat: isSortedSetMember * feat: zincrby * feat: sortedSetLex * feat: processSortedSet * fix: add mising await * feat: debug psql * fix: psql test * fix: test * fix: another test * fix: test fix * fix: psql tests * feat: remove logs * feat: user arrow func use builtin async promises * feat: topic bookmarks * feat: topic.delete * feat: topic.restore * feat: topics.purge * feat: merge * feat: suggested * feat: topics/user.js * feat: topics modules * feat: topics/follow * fix: deprecation msg * feat: fork * feat: topics/posts * feat: sorted/recent * feat: topic/teaser * feat: topics/tools * feat: topics/unread * feat: add back node versions disable deprecation notice wrap async controllers in try/catch * feat: use db directly * feat: promisify in place * fix: redis/psql * feat: deprecation message logs for psql * feat: more logs * feat: more logs * feat: logs again * feat: more logs * fix: call release * feat: restore travis, remove logs * fix: loops * feat: remove .async. usage
2019-07-09 12:46:49 -04:00
search.async = require('./promisify')(search);