| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const validator = require('validator'); | 
					
						
							|  |  |  | const _ = require('lodash'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const utils = require('../utils'); | 
					
						
							| 
									
										
										
										
											2021-01-18 13:32:55 -05:00
										 |  |  | const user = require('../user'); | 
					
						
							| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | const posts = require('../posts'); | 
					
						
							| 
									
										
										
										
											2020-10-17 21:24:33 -04:00
										 |  |  | const topics = require('../topics'); | 
					
						
							| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | const groups = require('../groups'); | 
					
						
							|  |  |  | const meta = require('../meta'); | 
					
						
							|  |  |  | const events = require('../events'); | 
					
						
							| 
									
										
										
										
											2020-10-17 21:24:33 -04:00
										 |  |  | const privileges = require('../privileges'); | 
					
						
							| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | const apiHelpers = require('./helpers'); | 
					
						
							|  |  |  | const websockets = require('../socket.io'); | 
					
						
							| 
									
										
										
										
											2021-01-18 15:31:14 -05:00
										 |  |  | const socketHelpers = require('../socket.io/helpers'); | 
					
						
							| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | const postsAPI = module.exports; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-28 19:38:00 -05:00
										 |  |  | postsAPI.get = async function (caller, data) { | 
					
						
							|  |  |  | 	const [userPrivileges, post, voted] = await Promise.all([ | 
					
						
							|  |  |  | 		privileges.posts.get([data.pid], caller.uid), | 
					
						
							|  |  |  | 		posts.getPostData(data.pid), | 
					
						
							|  |  |  | 		posts.hasVoted(data.pid, caller.uid), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 	if (!post) { | 
					
						
							|  |  |  | 		return null; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	Object.assign(post, voted); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const userPrivilege = userPrivileges[0]; | 
					
						
							|  |  |  | 	if (!userPrivilege.read || !userPrivilege['topics:read']) { | 
					
						
							|  |  |  | 		return null; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	post.ip = userPrivilege.isAdminOrMod ? post.ip : undefined; | 
					
						
							|  |  |  | 	const selfPost = caller.uid && caller.uid === parseInt(post.uid, 10); | 
					
						
							|  |  |  | 	if (post.deleted && !(userPrivilege.isAdminOrMod || selfPost)) { | 
					
						
							|  |  |  | 		post.content = '[[topic:post_is_deleted]]'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return post; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | postsAPI.edit = async function (caller, data) { | 
					
						
							|  |  |  | 	if (!data || !data.pid || (meta.config.minimumPostLength !== 0 && !data.content)) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Trim and remove HTML (latter for composers that send in HTML, like redactor)
 | 
					
						
							|  |  |  | 	const contentLen = utils.stripHTMLTags(data.content).trim().length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (data.title && data.title.length < meta.config.minimumTitleLength) { | 
					
						
							|  |  |  | 		throw new Error('[[error:title-too-short, ' + meta.config.minimumTitleLength + ']]'); | 
					
						
							|  |  |  | 	} else if (data.title && data.title.length > meta.config.maximumTitleLength) { | 
					
						
							|  |  |  | 		throw new Error('[[error:title-too-long, ' + meta.config.maximumTitleLength + ']]'); | 
					
						
							|  |  |  | 	} else if (meta.config.minimumPostLength !== 0 && contentLen < meta.config.minimumPostLength) { | 
					
						
							|  |  |  | 		throw new Error('[[error:content-too-short, ' + meta.config.minimumPostLength + ']]'); | 
					
						
							|  |  |  | 	} else if (contentLen > meta.config.maximumPostLength) { | 
					
						
							|  |  |  | 		throw new Error('[[error:content-too-long, ' + meta.config.maximumPostLength + ']]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	data.uid = caller.uid; | 
					
						
							|  |  |  | 	data.req = apiHelpers.buildReqObject(caller); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const editResult = await posts.edit(data); | 
					
						
							| 
									
										
										
										
											2020-12-01 10:37:42 -05:00
										 |  |  | 	if (editResult.topic.isMainPost) { | 
					
						
							| 
									
										
										
										
											2020-12-04 09:37:50 -05:00
										 |  |  | 		await topics.thumbs.migrate(data.uuid, editResult.topic.tid); | 
					
						
							| 
									
										
										
										
											2020-12-01 10:37:42 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | 	if (editResult.topic.renamed) { | 
					
						
							|  |  |  | 		await events.log({ | 
					
						
							|  |  |  | 			type: 'topic-rename', | 
					
						
							|  |  |  | 			uid: caller.uid, | 
					
						
							|  |  |  | 			ip: caller.ip, | 
					
						
							|  |  |  | 			tid: editResult.topic.tid, | 
					
						
							|  |  |  | 			oldTitle: validator.escape(String(editResult.topic.oldTitle)), | 
					
						
							|  |  |  | 			newTitle: validator.escape(String(editResult.topic.title)), | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const postObj = await posts.getPostSummaryByPids([editResult.post.pid], caller.uid, {}); | 
					
						
							|  |  |  | 	const returnData = { ...postObj[0], ...editResult.post }; | 
					
						
							| 
									
										
										
										
											2020-10-27 08:04:07 -04:00
										 |  |  | 	returnData.topic = { ...postObj[0].topic, ...editResult.post.topic }; | 
					
						
							| 
									
										
										
										
											2020-10-17 15:07:04 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!editResult.post.deleted) { | 
					
						
							|  |  |  | 		websockets.in('topic_' + editResult.topic.tid).emit('event:post_edited', editResult); | 
					
						
							|  |  |  | 		return returnData; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const memberData = await groups.getMembersOfGroups([ | 
					
						
							|  |  |  | 		'administrators', | 
					
						
							|  |  |  | 		'Global Moderators', | 
					
						
							|  |  |  | 		'cid:' + editResult.topic.cid + ':privileges:moderate', | 
					
						
							|  |  |  | 		'cid:' + editResult.topic.cid + ':privileges:groups:moderate', | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const uids = _.uniq(_.flatten(memberData).concat(String(caller.uid))); | 
					
						
							|  |  |  | 	uids.forEach(uid =>	websockets.in('uid_' + uid).emit('event:post_edited', editResult)); | 
					
						
							|  |  |  | 	return returnData; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-10-17 21:24:33 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | postsAPI.delete = async function (caller, data) { | 
					
						
							|  |  |  | 	await deleteOrRestore(caller, data, { | 
					
						
							|  |  |  | 		command: 'delete', | 
					
						
							|  |  |  | 		event: 'event:post_deleted', | 
					
						
							|  |  |  | 		type: 'post-delete', | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.restore = async function (caller, data) { | 
					
						
							|  |  |  | 	await deleteOrRestore(caller, data, { | 
					
						
							|  |  |  | 		command: 'restore', | 
					
						
							|  |  |  | 		event: 'event:post_restored', | 
					
						
							|  |  |  | 		type: 'post-restore', | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function deleteOrRestore(caller, data, params) { | 
					
						
							|  |  |  | 	if (!data || !data.pid) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const postData = await posts.tools[params.command](caller.uid, data.pid); | 
					
						
							|  |  |  | 	const results = await isMainAndLastPost(data.pid); | 
					
						
							|  |  |  | 	if (results.isMain && results.isLast) { | 
					
						
							|  |  |  | 		await deleteOrRestoreTopicOf(params.command, data.pid, caller); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	websockets.in('topic_' + postData.tid).emit(params.event, postData); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	await events.log({ | 
					
						
							|  |  |  | 		type: params.type, | 
					
						
							|  |  |  | 		uid: caller.uid, | 
					
						
							|  |  |  | 		pid: data.pid, | 
					
						
							|  |  |  | 		tid: postData.tid, | 
					
						
							|  |  |  | 		ip: caller.ip, | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function deleteOrRestoreTopicOf(command, pid, caller) { | 
					
						
							|  |  |  | 	const topic = await posts.getTopicFields(pid, ['tid', 'cid', 'deleted']); | 
					
						
							|  |  |  | 	// command: delete/restore
 | 
					
						
							|  |  |  | 	await apiHelpers.doTopicAction(command, | 
					
						
							|  |  |  | 		topic.deleted ? 'event:topic_restored' : 'event:topic_deleted', | 
					
						
							|  |  |  | 		caller, | 
					
						
							|  |  |  | 		{ tids: [topic.tid], cid: topic.cid } | 
					
						
							|  |  |  | 	); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.purge = async function (caller, data) { | 
					
						
							|  |  |  | 	if (!data || !parseInt(data.pid, 10)) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const results = await isMainAndLastPost(data.pid); | 
					
						
							|  |  |  | 	if (results.isMain && !results.isLast) { | 
					
						
							|  |  |  | 		throw new Error('[[error:cant-purge-main-post]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const isMainAndLast = results.isMain && results.isLast; | 
					
						
							|  |  |  | 	const postData = await posts.getPostFields(data.pid, ['toPid', 'tid']); | 
					
						
							|  |  |  | 	postData.pid = data.pid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const canPurge = await privileges.posts.canPurge(data.pid, caller.uid); | 
					
						
							|  |  |  | 	if (!canPurge) { | 
					
						
							|  |  |  | 		throw new Error('[[error:no-privileges]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	require('../posts/cache').del(data.pid); | 
					
						
							|  |  |  | 	await posts.purge(data.pid, caller.uid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	websockets.in('topic_' + postData.tid).emit('event:post_purged', postData); | 
					
						
							|  |  |  | 	const topicData = await topics.getTopicFields(postData.tid, ['title', 'cid']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	await events.log({ | 
					
						
							|  |  |  | 		type: 'post-purge', | 
					
						
							|  |  |  | 		pid: data.pid, | 
					
						
							|  |  |  | 		uid: caller.uid, | 
					
						
							|  |  |  | 		ip: caller.ip, | 
					
						
							|  |  |  | 		tid: postData.tid, | 
					
						
							|  |  |  | 		title: String(topicData.title), | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (isMainAndLast) { | 
					
						
							|  |  |  | 		await apiHelpers.doTopicAction('purge', 'event:topic_purged', | 
					
						
							|  |  |  | 			caller, | 
					
						
							|  |  |  | 			{ tids: [postData.tid], cid: topicData.cid } | 
					
						
							|  |  |  | 		); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function isMainAndLastPost(pid) { | 
					
						
							|  |  |  | 	const [isMain, topicData] = await Promise.all([ | 
					
						
							|  |  |  | 		posts.isMain(pid), | 
					
						
							|  |  |  | 		posts.getTopicFields(pid, ['postcount']), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 	return { | 
					
						
							|  |  |  | 		isMain: isMain, | 
					
						
							|  |  |  | 		isLast: topicData && topicData.postcount === 1, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-18 15:31:14 -05:00
										 |  |  | postsAPI.move = async function (caller, data) { | 
					
						
							|  |  |  | 	const canMove = await Promise.all([ | 
					
						
							|  |  |  | 		privileges.topics.isAdminOrMod(data.tid, caller.uid), | 
					
						
							|  |  |  | 		privileges.posts.canMove(data.pid, caller.uid), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 	if (!canMove.every(Boolean)) { | 
					
						
							|  |  |  | 		throw new Error('[[error:no-privileges]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	await topics.movePostToTopic(caller.uid, data.pid, data.tid); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const [postDeleted, topicDeleted] = await Promise.all([ | 
					
						
							|  |  |  | 		posts.getPostField(data.pid, 'deleted'), | 
					
						
							|  |  |  | 		topics.getTopicField(data.tid, 'deleted'), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!postDeleted && !topicDeleted) { | 
					
						
							|  |  |  | 		socketHelpers.sendNotificationToPostOwner(data.pid, caller.uid, 'move', 'notifications:moved_your_post'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-17 22:59:12 -04:00
										 |  |  | postsAPI.upvote = async function (caller, data) { | 
					
						
							|  |  |  | 	return await apiHelpers.postCommand(caller, 'upvote', 'voted', 'notifications:upvoted_your_post_in', data); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.downvote = async function (caller, data) { | 
					
						
							|  |  |  | 	return await apiHelpers.postCommand(caller, 'downvote', 'voted', '', data); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.unvote = async function (caller, data) { | 
					
						
							|  |  |  | 	return await apiHelpers.postCommand(caller, 'unvote', 'voted', '', data); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.bookmark = async function (caller, data) { | 
					
						
							|  |  |  | 	return await apiHelpers.postCommand(caller, 'bookmark', 'bookmarked', '', data); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.unbookmark = async function (caller, data) { | 
					
						
							|  |  |  | 	return await apiHelpers.postCommand(caller, 'unbookmark', 'bookmarked', '', data); | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2021-01-18 13:32:55 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | async function diffsPrivilegeCheck(pid, uid) { | 
					
						
							|  |  |  | 	const [deleted, privilegesData] = await Promise.all([ | 
					
						
							|  |  |  | 		posts.getPostField(pid, 'deleted'), | 
					
						
							|  |  |  | 		privileges.posts.get([pid], uid), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const allowed = privilegesData[0]['posts:history'] && (deleted ? privilegesData[0]['posts:view_deleted'] : true); | 
					
						
							|  |  |  | 	if (!allowed) { | 
					
						
							|  |  |  | 		throw new Error('[[error:no-privileges]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.getDiffs = async (caller, data) => { | 
					
						
							|  |  |  | 	await diffsPrivilegeCheck(data.pid, caller.uid); | 
					
						
							|  |  |  | 	const timestamps = await posts.diffs.list(data.pid); | 
					
						
							|  |  |  | 	const post = await posts.getPostFields(data.pid, ['timestamp', 'uid']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const diffs = await posts.diffs.get(data.pid); | 
					
						
							|  |  |  | 	const uids = diffs.map(diff => diff.uid || null); | 
					
						
							|  |  |  | 	uids.push(post.uid); | 
					
						
							|  |  |  | 	let usernames = await user.getUsersFields(uids, ['username']); | 
					
						
							|  |  |  | 	usernames = usernames.map(userObj => (userObj.uid ? userObj.username : null)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	let canEdit = true; | 
					
						
							|  |  |  | 	try { | 
					
						
							|  |  |  | 		await user.isPrivilegedOrSelf(caller.uid, post.uid); | 
					
						
							|  |  |  | 	} catch (e) { | 
					
						
							|  |  |  | 		canEdit = false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	timestamps.push(post.timestamp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return { | 
					
						
							|  |  |  | 		timestamps: timestamps, | 
					
						
							|  |  |  | 		revisions: timestamps.map((timestamp, idx) => ({ | 
					
						
							|  |  |  | 			timestamp: timestamp, | 
					
						
							|  |  |  | 			username: usernames[idx], | 
					
						
							|  |  |  | 		})), | 
					
						
							|  |  |  | 		editable: canEdit, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.loadDiff = async (caller, data) => { | 
					
						
							|  |  |  | 	await diffsPrivilegeCheck(data.pid, caller.uid); | 
					
						
							|  |  |  | 	return await posts.diffs.load(data.pid, data.since, caller.uid); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | postsAPI.restoreDiff = async (caller, data) => { | 
					
						
							|  |  |  | 	const cid = await posts.getCidByPid(data.pid); | 
					
						
							|  |  |  | 	const canEdit = await privileges.categories.can('edit', cid, caller.uid); | 
					
						
							|  |  |  | 	if (!canEdit) { | 
					
						
							|  |  |  | 		throw new Error('[[error:no-privileges]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const edit = await posts.diffs.restore(data.pid, data.since, caller.uid, apiHelpers.buildReqObject(caller)); | 
					
						
							|  |  |  | 	websockets.in('topic_' + edit.topic.tid).emit('event:post_edited', edit); | 
					
						
							|  |  |  | }; |