You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
nodebb/src/search.js

440 lines
11 KiB
JavaScript

11 years ago
'use strict';
9 years ago
var async = require('async');
8 years ago
var _ = require('lodash');
9 years ago
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');
11 years ago
8 years ago
var search = module.exports;
11 years ago
search.search = function (data, callback) {
11 years ago
var start = process.hrtime();
var searchIn = data.searchIn || 'titlesposts';
11 years ago
9 years ago
async.waterfall([
function (next) {
if (searchIn === 'posts' || searchIn === 'titles' || searchIn === 'titlesposts') {
searchInContent(data, next);
} else if (searchIn === 'users') {
user.search(data, next);
} else if (searchIn === 'tags') {
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);
},
9 years ago
], callback);
10 years ago
};
function searchInContent(data, callback) {
data.uid = data.uid || 0;
8 years ago
var matchCount = 0;
var pids;
var metadata;
var itemsPerPage = data.itemsPerPage || 10;
8 years ago
async.waterfall([
function (next) {
async.parallel({
searchCids: function (next) {
getSearchCids(data, next);
},
searchUids: function (next) {
getSearchUids(data, next);
},
}, next);
},
8 years ago
function (results, next) {
function doSearch(type, searchIn, next) {
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,
7 years ago
searchData: data,
}, next);
} else {
next(null, []);
}
}
8 years ago
async.parallel({
pids: function (next) {
doSearch('post', ['posts', 'titlesposts'], next);
},
tids: function (next) {
doSearch('topic', ['titles', 'titlesposts'], next);
},
}, next);
},
function (results, next) {
pids = results.pids;
if (!results || (!results.pids.length && !results.tids.length)) {
return callback(null, { posts: [], matchCount: matchCount, pageCount: 1 });
10 years ago
}
8 years ago
topics.getMainPids(results.tids, next);
},
function (mainPids, next) {
pids = mainPids.concat(pids).map(function (pid) {
return pid && pid.toString();
8 years ago
}).filter(Boolean);
8 years ago
privileges.posts.filter('read', pids, data.uid, next);
},
function (pids, next) {
filterAndSort(pids, data, next);
},
function (pids, next) {
plugins.fireHook('filter:search.inContent', {
pids: pids,
}, next);
},
function (_metadata, next) {
metadata = _metadata;
matchCount = metadata.pids.length;
8 years ago
if (data.page) {
var start = Math.max(0, (data.page - 1)) * itemsPerPage;
metadata.pids = metadata.pids.slice(start, start + itemsPerPage);
8 years ago
}
posts.getPostSummaryByPids(metadata.pids, data.uid, {}, next);
8 years ago
},
function (posts, next) {
// Append metadata to returned payload (without pids)
delete metadata.pids;
next(null, Object.assign({
posts: posts,
matchCount: matchCount,
pageCount: Math.max(1, Math.ceil(parseInt(matchCount, 10) / 10)),
}, metadata));
8 years ago
},
], callback);
10 years ago
}
11 years ago
function filterAndSort(pids, data, callback) {
8 years ago
async.waterfall([
function (next) {
getMatchedPosts(pids, data, next);
},
function (posts, next) {
if (!posts.length) {
8 years ago
return callback(null, pids);
}
posts = posts.filter(Boolean);
8 years ago
posts = filterByPostcount(posts, data.replies, data.repliesFilter);
posts = filterByTimerange(posts, data.timeRange, data.timeFilter);
posts = filterByTags(posts, data.hasTags);
8 years ago
sortPosts(posts, data);
plugins.fireHook('filter:search.filterAndSort', { pids: pids, posts: posts, data: data }, next);
},
function (result, next) {
pids = result.posts.map(function (post) {
8 years ago
return post && post.pid;
});
8 years ago
next(null, pids);
},
], callback);
}
function getMatchedPosts(pids, data, callback) {
var postFields = ['pid', 'uid', 'tid', 'timestamp', 'deleted'];
var categoryFields = [];
8 years ago
if (data.sortBy && data.sortBy !== 'relevance') {
if (data.sortBy.startsWith('category.')) {
categoryFields.push(data.sortBy.split('.')[1]);
}
}
var posts;
async.waterfall([
function (next) {
var keys = pids.map(function (pid) {
return 'post:' + pid;
});
db.getObjectsFields(keys, postFields, next);
},
function (_posts, next) {
posts = _posts.filter(function (post) {
10 years ago
return post && parseInt(post.deleted, 10) !== 1;
});
async.parallel({
users: function (next) {
if (data.sortBy && data.sortBy.startsWith('user')) {
var uids = posts.map(function (post) {
return post.uid;
});
9 years ago
user.getUsersFields(uids, ['username'], next);
} else {
next();
}
},
topics: function (next) {
var topicsData;
async.waterfall([
function (next) {
var topicKeys = posts.map(function (post) {
return 'topic:' + post.tid;
});
db.getObjects(topicKeys, next);
},
function (_topics, next) {
topicsData = _topics;
async.parallel({
teasers: function (next) {
8 years ago
if (data.sortBy && data.sortBy.startsWith('teaser')) {
var teaserKeys = topicsData.map(function (topic) {
return 'post:' + topic.teaserPid;
});
db.getObjectsFields(teaserKeys, ['timestamp'], next);
} else {
next();
}
},
categories: function (next) {
if (!categoryFields.length) {
return next();
}
var cids = topicsData.map(function (topic) {
return 'category:' + topic.cid;
});
db.getObjectsFields(cids, categoryFields, next);
},
tags: function (next) {
7 years ago
if (Array.isArray(data.hasTags) && data.hasTags.length) {
var tids = posts.map(function (post) {
return post && post.tid;
});
topics.getTopicsTags(tids, next);
} else {
setImmediate(next);
}
},
}, next);
},
8 years ago
function (results, next) {
topicsData.forEach(function (topic, index) {
if (topic && results.categories && results.categories[index]) {
topic.category = results.categories[index];
}
if (topic && results.teasers && results.teasers[index]) {
topic.teaser = results.teasers[index];
}
if (topic && results.tags && results.tags[index]) {
topic.tags = results.tags[index];
}
});
8 years ago
next(null, topicsData);
},
], next);
},
}, next);
},
function (results, next) {
posts.forEach(function (post, index) {
if (results.topics && results.topics[index]) {
post.topic = results.topics[index];
if (results.topics[index].category) {
post.category = results.topics[index].category;
}
if (results.topics[index].teaser) {
post.teaser = results.topics[index].teaser;
}
}
if (results.users && results.users[index]) {
post.user = results.users[index];
}
});
posts = posts.filter(function (post) {
10 years ago
return post && post.topic && parseInt(post.topic.deleted, 10) !== 1;
});
next(null, posts);
},
], callback);
}
function filterByPostcount(posts, postCount, repliesFilter) {
postCount = parseInt(postCount, 10);
if (postCount) {
if (repliesFilter === 'atleast') {
posts = posts.filter(function (post) {
return post.topic && post.topic.postcount >= postCount;
});
} else {
posts = posts.filter(function (post) {
return post.topic && post.topic.postcount <= postCount;
});
}
}
return posts;
}
function filterByTimerange(posts, timeRange, timeFilter) {
8 years ago
timeRange = parseInt(timeRange, 10) * 1000;
if (timeRange) {
var time = Date.now() - timeRange;
if (timeFilter === 'newer') {
posts = posts.filter(function (post) {
return post.timestamp >= time;
});
} else {
posts = posts.filter(function (post) {
return post.timestamp <= time;
});
}
}
return posts;
}
function filterByTags(posts, hasTags) {
7 years ago
if (Array.isArray(hasTags) && hasTags.length) {
posts = posts.filter(function (post) {
var hasAllTags = false;
7 years ago
if (post && post.topic && Array.isArray(post.topic.tags) && post.topic.tags.length) {
hasAllTags = hasTags.every(function (tag) {
return post.topic.tags.indexOf(tag) !== -1;
});
}
return hasAllTags;
});
}
return posts;
}
function sortPosts(posts, data) {
8 years ago
if (!posts.length || !data.sortBy || data.sortBy === 'relevance') {
return;
}
data.sortDirection = data.sortDirection || 'desc';
var direction = data.sortDirection === 'desc' ? 1 : -1;
if (data.sortBy === 'timestamp') {
posts.sort(function (p1, p2) {
return direction * (p2.timestamp - p1.timestamp);
});
return;
}
var firstPost = posts[0];
var fields = data.sortBy.split('.');
if (!fields || fields.length !== 2 || !firstPost[fields[0]] || !firstPost[fields[0]][fields[1]]) {
return;
}
var isNumeric = utils.isNumber(firstPost[fields[0]][fields[1]]);
if (isNumeric) {
posts.sort(function (p1, p2) {
return direction * (p2[fields[0]][fields[1]] - p1[fields[0]][fields[1]]);
});
} else {
posts.sort(function (p1, p2) {
9 years ago
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;
});
}
}
function getSearchCids(data, callback) {
if (!Array.isArray(data.categories) || !data.categories.length) {
return callback(null, []);
}
if (data.categories.indexOf('all') !== -1) {
async.waterfall([
function (next) {
db.getSortedSetRange('categories:cid', 0, -1, next);
},
function (cids, next) {
privileges.categories.filterCids('read', cids, data.uid, next);
},
], callback);
return;
}
8 years ago
async.waterfall([
function (next) {
async.parallel({
watchedCids: function (next) {
if (data.categories.indexOf('watched') !== -1) {
user.getWatchedCategories(data.uid, next);
} else {
next(null, []);
}
},
childrenCids: function (next) {
if (data.searchChildren) {
getChildrenCids(data.categories, data.uid, next);
} else {
next(null, []);
}
},
}, next);
},
8 years ago
function (results, next) {
8 years ago
var cids = results.watchedCids.concat(results.childrenCids).concat(data.categories).filter(Boolean);
cids = _.uniq(cids);
8 years ago
next(null, cids);
},
8 years ago
], callback);
}
function getChildrenCids(cids, uid, callback) {
8 years ago
async.waterfall([
function (next) {
categories.getChildren(cids, uid, next);
},
function (childrenCategories, next) {
var childrenCids = [];
var allCategories = [];
childrenCategories.forEach(function (childrens) {
categories.flattenCategories(allCategories, childrens);
childrenCids = childrenCids.concat(allCategories.map(function (category) {
return category && category.cid;
}));
});
8 years ago
next(null, childrenCids);
},
], callback);
}
function getSearchUids(data, callback) {
if (data.postedBy) {
user.getUidsByUsernames(Array.isArray(data.postedBy) ? data.postedBy : [data.postedBy], callback);
} else {
callback(null, []);
}
}