mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-26 16:46:12 +01:00 
			
		
		
		
	feat: #7743 , search.js
This commit is contained in:
		
							
								
								
									
										372
									
								
								src/search.js
									
									
									
									
									
								
							
							
						
						
									
										372
									
								
								src/search.js
									
									
									
									
									
								
							| @@ -1,107 +1,89 @@ | |||||||
| 'use strict'; | 'use strict'; | ||||||
|  |  | ||||||
| var async = require('async'); | const _ = require('lodash'); | ||||||
| var _ = require('lodash'); |  | ||||||
|  |  | ||||||
| var db = require('./database'); | const db = require('./database'); | ||||||
| var posts = require('./posts'); | const posts = require('./posts'); | ||||||
| var topics = require('./topics'); | const topics = require('./topics'); | ||||||
| var categories = require('./categories'); | const categories = require('./categories'); | ||||||
| var user = require('./user'); | const user = require('./user'); | ||||||
| var plugins = require('./plugins'); | const plugins = require('./plugins'); | ||||||
| var privileges = require('./privileges'); | const privileges = require('./privileges'); | ||||||
| var utils = require('./utils'); | const utils = require('./utils'); | ||||||
|  |  | ||||||
| var search = module.exports; | const search = module.exports; | ||||||
|  |  | ||||||
| search.search = function (data, callback) { | search.search = async function (data) { | ||||||
| 	var start = process.hrtime(); | 	const start = process.hrtime(); | ||||||
| 	data.searchIn = data.searchIn || 'titlesposts'; | 	data.searchIn = data.searchIn || 'titlesposts'; | ||||||
| 	data.sortBy = data.sortBy || 'relevance'; | 	data.sortBy = data.sortBy || 'relevance'; | ||||||
| 	async.waterfall([ |  | ||||||
| 		function (next) { | 	let result; | ||||||
| 	if (data.searchIn === 'posts' || data.searchIn === 'titles' || data.searchIn === 'titlesposts') { | 	if (data.searchIn === 'posts' || data.searchIn === 'titles' || data.searchIn === 'titlesposts') { | ||||||
| 				searchInContent(data, next); | 		result = await searchInContent(data); | ||||||
| 	} else if (data.searchIn === 'users') { | 	} else if (data.searchIn === 'users') { | ||||||
| 				user.search(data, next); | 		result = await user.search(data); | ||||||
| 	} else if (data.searchIn === 'tags') { | 	} else if (data.searchIn === 'tags') { | ||||||
| 				topics.searchAndLoadTags(data, next); | 		result = await topics.searchAndLoadTags(data); | ||||||
| 	} else { | 	} else { | ||||||
| 				next(new Error('[[error:unknown-search-filter]]')); | 		throw new Error('[[error:unknown-search-filter]]'); | ||||||
| 	} | 	} | ||||||
| 		}, |  | ||||||
| 		function (result, next) { |  | ||||||
| 	result.time = (process.elapsedTimeSince(start) / 1000).toFixed(2); | 	result.time = (process.elapsedTimeSince(start) / 1000).toFixed(2); | ||||||
| 			next(null, result); | 	return result; | ||||||
| 		}, |  | ||||||
| 	], callback); |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| function searchInContent(data, callback) { | async function searchInContent(data) { | ||||||
| 	data.uid = data.uid || 0; | 	data.uid = data.uid || 0; | ||||||
| 	var pids; |  | ||||||
| 	var metadata; | 	const itemsPerPage = Math.min(data.itemsPerPage || 10, 100); | ||||||
| 	var itemsPerPage = Math.min(data.itemsPerPage || 10, 100); |  | ||||||
| 	const returnData = { | 	const returnData = { | ||||||
| 		posts: [], | 		posts: [], | ||||||
| 		matchCount: 0, | 		matchCount: 0, | ||||||
| 		pageCount: 1, | 		pageCount: 1, | ||||||
| 	}; | 	}; | ||||||
| 	async.waterfall([ |  | ||||||
| 		function (next) { | 	const [searchCids, searchUids] = await Promise.all([ | ||||||
| 			async.parallel({ | 		getSearchCids(data), | ||||||
| 				searchCids: async.apply(getSearchCids, data), | 		getSearchUids(data), | ||||||
| 				searchUids: async.apply(getSearchUids, data), | 	]); | ||||||
| 			}, next); |  | ||||||
| 		}, | 	async function doSearch(type, searchIn) { | ||||||
| 		function (results, next) { |  | ||||||
| 			function doSearch(type, searchIn, next) { |  | ||||||
| 		if (searchIn.includes(data.searchIn)) { | 		if (searchIn.includes(data.searchIn)) { | ||||||
| 					plugins.fireHook('filter:search.query', { | 			return await plugins.fireHook('filter:search.query', { | ||||||
| 				index: type, | 				index: type, | ||||||
| 				content: data.query, | 				content: data.query, | ||||||
| 				matchWords: data.matchWords || 'all', | 				matchWords: data.matchWords || 'all', | ||||||
| 						cid: results.searchCids, | 				cid: searchCids, | ||||||
| 						uid: results.searchUids, | 				uid: searchUids, | ||||||
| 				searchData: data, | 				searchData: data, | ||||||
| 					}, next); | 			}); | ||||||
| 				} else { |  | ||||||
| 					next(null, []); |  | ||||||
| 		} | 		} | ||||||
|  | 		return []; | ||||||
| 	} | 	} | ||||||
| 			async.parallel({ | 	const [pids, tids] = await Promise.all([ | ||||||
| 				pids: async.apply(doSearch, 'post', ['posts', 'titlesposts']), | 		doSearch('post', ['posts', 'titlesposts']), | ||||||
| 				tids: async.apply(doSearch, 'topic', ['titles', 'titlesposts']), | 		doSearch('topic', ['titles', 'titlesposts']), | ||||||
| 			}, next); | 	]); | ||||||
| 		}, |  | ||||||
| 		function (results, next) { |  | ||||||
| 			pids = results.pids; |  | ||||||
|  |  | ||||||
| 	if (data.returnIds) { | 	if (data.returnIds) { | ||||||
| 				return callback(null, results); | 		return { pids: pids, tids: tids }; | ||||||
|  | 	} | ||||||
|  | 	if (!pids.length && !tids.length) { | ||||||
|  | 		return returnData; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 			if (!results.pids.length && !results.tids.length) { | 	const mainPids = await topics.getMainPids(tids); | ||||||
| 				return callback(null, returnData); |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			topics.getMainPids(results.tids, next); | 	let allPids = mainPids.concat(pids).filter(Boolean); | ||||||
| 		}, |  | ||||||
| 		function (mainPids, next) { | 	allPids = await privileges.posts.filter('topics:read', allPids, data.uid); | ||||||
| 			pids = mainPids.concat(pids).filter(Boolean); | 	allPids = await filterAndSort(allPids, data); | ||||||
|  |  | ||||||
|  | 	const metadata = await plugins.fireHook('filter:search.inContent', { | ||||||
|  | 		pids: allPids, | ||||||
|  | 	}); | ||||||
|  |  | ||||||
| 			privileges.posts.filter('topics: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; |  | ||||||
| 	returnData.matchCount = metadata.pids.length; | 	returnData.matchCount = metadata.pids.length; | ||||||
| 	returnData.pageCount = Math.max(1, Math.ceil(parseInt(returnData.matchCount, 10) / itemsPerPage)); | 	returnData.pageCount = Math.max(1, Math.ceil(parseInt(returnData.matchCount, 10) / itemsPerPage)); | ||||||
|  |  | ||||||
| @@ -110,124 +92,49 @@ function searchInContent(data, callback) { | |||||||
| 		metadata.pids = metadata.pids.slice(start, start + itemsPerPage); | 		metadata.pids = metadata.pids.slice(start, start + itemsPerPage); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 			posts.getPostSummaryByPids(metadata.pids, data.uid, {}, next); | 	returnData.posts = await posts.getPostSummaryByPids(metadata.pids, data.uid, {}); | ||||||
| 		}, |  | ||||||
| 		function (posts, next) { |  | ||||||
| 			returnData.posts = posts; |  | ||||||
| 			// Append metadata to returned payload (without pids) |  | ||||||
| 	delete metadata.pids; | 	delete metadata.pids; | ||||||
| 			next(null, Object.assign(returnData, metadata)); | 	return Object.assign(returnData, metadata); | ||||||
| 		}, |  | ||||||
| 	], callback); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function filterAndSort(pids, data, callback) { | async function filterAndSort(pids, data) { | ||||||
| 	if (data.sortBy === 'relevance' && !data.replies && !data.timeRange && !data.hasTags) { | 	if (data.sortBy === 'relevance' && !data.replies && !data.timeRange && !data.hasTags) { | ||||||
| 		return setImmediate(callback, null, pids); | 		return pids; | ||||||
| 	} | 	} | ||||||
|  | 	let postsData = await getMatchedPosts(pids, data); | ||||||
| 	async.waterfall([ | 	if (!postsData.length) { | ||||||
| 		function (next) { | 		return pids; | ||||||
| 			getMatchedPosts(pids, data, next); |  | ||||||
| 		}, |  | ||||||
| 		function (posts, next) { |  | ||||||
| 			if (!posts.length) { |  | ||||||
| 				return callback(null, pids); |  | ||||||
| 	} | 	} | ||||||
| 			posts = posts.filter(Boolean); | 	postsData = postsData.filter(Boolean); | ||||||
|  |  | ||||||
| 			posts = filterByPostcount(posts, data.replies, data.repliesFilter); | 	postsData = filterByPostcount(postsData, data.replies, data.repliesFilter); | ||||||
| 			posts = filterByTimerange(posts, data.timeRange, data.timeFilter); | 	postsData = filterByTimerange(postsData, data.timeRange, data.timeFilter); | ||||||
| 			posts = filterByTags(posts, data.hasTags); | 	postsData = filterByTags(postsData, data.hasTags); | ||||||
|  |  | ||||||
| 			sortPosts(posts, data); | 	sortPosts(postsData, data); | ||||||
|  |  | ||||||
| 			plugins.fireHook('filter:search.filterAndSort', { pids: pids, posts: posts, data: data }, next); | 	const result = await plugins.fireHook('filter:search.filterAndSort', { pids: pids, posts: postsData, data: data }); | ||||||
| 		}, |  | ||||||
| 		function (result, next) { | 	return result.posts.map(post => post && post.pid); | ||||||
| 			pids = result.posts.map(post => post && post.pid); |  | ||||||
| 			next(null, pids); |  | ||||||
| 		}, |  | ||||||
| 	], callback); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function getMatchedPosts(pids, data, callback) { | async function getMatchedPosts(pids, data) { | ||||||
| 	var postFields = ['pid', 'uid', 'tid', 'timestamp', 'deleted', 'upvotes', 'downvotes']; | 	const postFields = ['pid', 'uid', 'tid', 'timestamp', 'deleted', 'upvotes', 'downvotes']; | ||||||
| 	var categoryFields = []; |  | ||||||
|  |  | ||||||
| 	if (data.sortBy.startsWith('category.')) { | 	let postsData = await posts.getPostsFields(pids, postFields); | ||||||
| 		categoryFields.push(data.sortBy.split('.')[1]); | 	postsData = postsData.filter(post => post && !post.deleted); | ||||||
| 	} | 	const uids = _.uniq(postsData.map(post => post.uid)); | ||||||
|  | 	const tids = _.uniq(postsData.map(post => post.tid)); | ||||||
|  |  | ||||||
| 	var postsData; | 	const [users, topics] = await Promise.all([ | ||||||
| 	let tids; | 		getUsers(uids, data), | ||||||
| 	let uids; | 		getTopics(tids, data), | ||||||
| 	async.waterfall([ | 	]); | ||||||
| 		function (next) { |  | ||||||
| 			posts.getPostsFields(pids, postFields, next); |  | ||||||
| 		}, |  | ||||||
| 		function (_postsData, next) { |  | ||||||
| 			postsData = _postsData.filter(post => post && !post.deleted); |  | ||||||
|  |  | ||||||
| 			async.parallel({ | 	const tidToTopic = _.zipObject(tids, topics); | ||||||
| 				users: function (next) { | 	const uidToUser = _.zipObject(uids, users); | ||||||
| 					if (data.sortBy.startsWith('user')) { |  | ||||||
| 						uids = _.uniq(postsData.map(post => post.uid)); |  | ||||||
| 						user.getUsersFields(uids, ['username'], next); |  | ||||||
| 					} else { |  | ||||||
| 						next(); |  | ||||||
| 					} |  | ||||||
| 				}, |  | ||||||
| 				topics: function (next) { |  | ||||||
| 					var topicsData; |  | ||||||
| 					tids = _.uniq(postsData.map(post => post.tid)); |  | ||||||
| 					let cids; |  | ||||||
| 					async.waterfall([ |  | ||||||
| 						function (next) { |  | ||||||
| 							topics.getTopicsData(tids, next); |  | ||||||
| 						}, |  | ||||||
| 						function (_topics, next) { |  | ||||||
| 							topicsData = _topics; |  | ||||||
| 							async.parallel({ |  | ||||||
| 								categories: function (next) { |  | ||||||
| 									if (!categoryFields.length) { |  | ||||||
| 										return next(); |  | ||||||
| 									} |  | ||||||
|  |  | ||||||
| 									cids = _.uniq(topicsData.map(topic => topic && topic.cid)); |  | ||||||
| 									db.getObjectsFields(cids.map(cid => 'category:' + cid), categoryFields, next); |  | ||||||
| 								}, |  | ||||||
| 								tags: function (next) { |  | ||||||
| 									if (Array.isArray(data.hasTags) && data.hasTags.length) { |  | ||||||
| 										topics.getTopicsTags(tids, next); |  | ||||||
| 									} else { |  | ||||||
| 										setImmediate(next); |  | ||||||
| 									} |  | ||||||
| 								}, |  | ||||||
| 							}, next); |  | ||||||
| 						}, |  | ||||||
| 						function (results, next) { |  | ||||||
| 							const cidToCategory = _.zipObject(cids, results.categories); |  | ||||||
| 							topicsData.forEach(function (topic, index) { |  | ||||||
| 								if (topic && results.categories && cidToCategory[topic.cid]) { |  | ||||||
| 									topic.category = cidToCategory[topic.cid]; |  | ||||||
| 								} |  | ||||||
| 								if (topic && results.tags && results.tags[index]) { |  | ||||||
| 									topic.tags = results.tags[index]; |  | ||||||
| 								} |  | ||||||
| 							}); |  | ||||||
|  |  | ||||||
| 							next(null, topicsData); |  | ||||||
| 						}, |  | ||||||
| 					], next); |  | ||||||
| 				}, |  | ||||||
| 			}, next); |  | ||||||
| 		}, |  | ||||||
| 		function (results, next) { |  | ||||||
| 			const tidToTopic = _.zipObject(tids, results.topics); |  | ||||||
| 			const uidToUser = _.zipObject(uids, results.users); |  | ||||||
| 	postsData.forEach(function (post) { | 	postsData.forEach(function (post) { | ||||||
| 				if (results.topics && tidToTopic[post.tid]) { | 		if (topics && tidToTopic[post.tid]) { | ||||||
| 			post.topic = tidToTopic[post.tid]; | 			post.topic = tidToTopic[post.tid]; | ||||||
| 			if (post.topic && post.topic.category) { | 			if (post.topic && post.topic.category) { | ||||||
| 				post.category = post.topic.category; | 				post.category = post.topic.category; | ||||||
| @@ -239,10 +146,55 @@ function getMatchedPosts(pids, data, callback) { | |||||||
| 		} | 		} | ||||||
| 	}); | 	}); | ||||||
|  |  | ||||||
| 			postsData = postsData.filter(post => post && post.topic && !post.topic.deleted); | 	return postsData.filter(post => post && post.topic && !post.topic.deleted); | ||||||
| 			next(null, postsData); | } | ||||||
| 		}, |  | ||||||
| 	], callback); | async function getUsers(uids, data) { | ||||||
|  | 	if (data.sortBy.startsWith('user')) { | ||||||
|  | 		return user.getUsersFields(uids, ['username']); | ||||||
|  | 	} | ||||||
|  | 	return []; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function getTopics(tids, data) { | ||||||
|  | 	const topicsData = await topics.getTopicsData(tids); | ||||||
|  | 	const cids = _.uniq(topicsData.map(topic => topic && topic.cid)); | ||||||
|  | 	const [categories, tags] = await Promise.all([ | ||||||
|  | 		getCategories(topicsData, data), | ||||||
|  | 		getTags(tids, data), | ||||||
|  | 	]); | ||||||
|  |  | ||||||
|  | 	const cidToCategory = _.zipObject(cids, categories); | ||||||
|  | 	topicsData.forEach(function (topic, index) { | ||||||
|  | 		if (topic && categories && cidToCategory[topic.cid]) { | ||||||
|  | 			topic.category = cidToCategory[topic.cid]; | ||||||
|  | 		} | ||||||
|  | 		if (topic && tags && tags[index]) { | ||||||
|  | 			topic.tags = tags[index]; | ||||||
|  | 		} | ||||||
|  | 	}); | ||||||
|  |  | ||||||
|  | 	return topicsData; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function getCategories(cids, data) { | ||||||
|  | 	const categoryFields = []; | ||||||
|  |  | ||||||
|  | 	if (data.sortBy.startsWith('category.')) { | ||||||
|  | 		categoryFields.push(data.sortBy.split('.')[1]); | ||||||
|  | 	} | ||||||
|  | 	if (!categoryFields.length) { | ||||||
|  | 		return null; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return await db.getObjectsFields(cids.map(cid => 'category:' + cid), categoryFields); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function getTags(tids, data) { | ||||||
|  | 	if (Array.isArray(data.hasTags) && data.hasTags.length) { | ||||||
|  | 		return await topics.getTopicsTags(tids); | ||||||
|  | 	} | ||||||
|  | 	return null; | ||||||
| } | } | ||||||
|  |  | ||||||
| function filterByPostcount(posts, postCount, repliesFilter) { | function filterByPostcount(posts, postCount, repliesFilter) { | ||||||
| @@ -316,58 +268,42 @@ function sortPosts(posts, data) { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| function getSearchCids(data, callback) { | async function getSearchCids(data) { | ||||||
| 	if (!Array.isArray(data.categories) || !data.categories.length) { | 	if (!Array.isArray(data.categories) || !data.categories.length) { | ||||||
| 		return callback(null, []); | 		return []; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (data.categories.includes('all')) { | 	if (data.categories.includes('all')) { | ||||||
| 		return categories.getCidsByPrivilege('categories:cid', data.uid, 'read', callback); | 		return await categories.getCidsByPrivilege('categories:cid', data.uid, 'read'); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	async.waterfall([ | 	const [watchedCids, childrenCids] = await Promise.all([ | ||||||
| 		function (next) { | 		getWatchedCids(data), | ||||||
| 			async.parallel({ | 		getChildrenCids(data), | ||||||
| 				watchedCids: function (next) { | 	]); | ||||||
| 					if (data.categories.includes('watched')) { | 	return _.uniq(watchedCids.concat(childrenCids).concat(data.categories).filter(Boolean)); | ||||||
| 						user.getCategoriesByStates(data.uid, [categories.watchStates.watching], next); |  | ||||||
| 					} else { |  | ||||||
| 						setImmediate(next, null, []); |  | ||||||
| 					} |  | ||||||
| 				}, |  | ||||||
| 				childrenCids: function (next) { |  | ||||||
| 					if (data.searchChildren) { |  | ||||||
| 						getChildrenCids(data.categories, data.uid, next); |  | ||||||
| 					} else { |  | ||||||
| 						setImmediate(next, null, []); |  | ||||||
| 					} |  | ||||||
| 				}, |  | ||||||
| 			}, next); |  | ||||||
| 		}, |  | ||||||
| 		function (results, next) { |  | ||||||
| 			const cids = _.uniq(results.watchedCids.concat(results.childrenCids).concat(data.categories).filter(Boolean)); |  | ||||||
| 			next(null, cids); |  | ||||||
| 		}, |  | ||||||
| 	], callback); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function getChildrenCids(cids, uid, callback) { | async function getWatchedCids(data) { | ||||||
| 	async.waterfall([ | 	if (!data.categories.includes('watched')) { | ||||||
| 		function (next) { | 		return []; | ||||||
| 			async.map(cids, categories.getChildrenCids, next); | 	} | ||||||
| 		}, | 	return await user.getCategoriesByStates(data.uid, [categories.watchStates.watching]); | ||||||
| 		function (childrenCids, next) { |  | ||||||
| 			privileges.categories.filterCids('find', _.uniq(_.flatten(childrenCids)), uid, next); |  | ||||||
| 		}, |  | ||||||
| 	], callback); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function getSearchUids(data, callback) { | async function getChildrenCids(data) { | ||||||
| 	if (data.postedBy) { | 	if (!data.searchChildren) { | ||||||
| 		user.getUidsByUsernames(Array.isArray(data.postedBy) ? data.postedBy : [data.postedBy], callback); | 		return []; | ||||||
| 	} else { |  | ||||||
| 		setImmediate(callback, null, []); |  | ||||||
| 	} | 	} | ||||||
|  | 	const childrenCids = await Promise.all(data.categories.map(cid => categories.getChildrenCids(cid))); | ||||||
|  | 	return await privileges.categories.filterCids('find', _.uniq(_.flatten(childrenCids)), data.uid); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function getSearchUids(data) { | ||||||
|  | 	if (!data.postedBy) { | ||||||
|  | 		return []; | ||||||
|  | 	} | ||||||
|  | 	return await user.getUidsByUsernames(Array.isArray(data.postedBy) ? data.postedBy : [data.postedBy]); | ||||||
| } | } | ||||||
|  |  | ||||||
| search.async = require('./promisify')(search); | search.async = require('./promisify')(search); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user