| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | const _ = require('lodash'); | 
					
						
							|  |  |  | const winston = require('winston'); | 
					
						
							|  |  |  | const validator = require('validator'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const db = require('./database'); | 
					
						
							|  |  |  | const user = require('./user'); | 
					
						
							|  |  |  | const groups = require('./groups'); | 
					
						
							|  |  |  | const meta = require('./meta'); | 
					
						
							|  |  |  | const notifications = require('./notifications'); | 
					
						
							|  |  |  | const analytics = require('./analytics'); | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | const categories = require('./categories'); | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | const topics = require('./topics'); | 
					
						
							|  |  |  | const posts = require('./posts'); | 
					
						
							|  |  |  | const privileges = require('./privileges'); | 
					
						
							|  |  |  | const plugins = require('./plugins'); | 
					
						
							|  |  |  | const utils = require('../public/src/utils'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const Flags = module.exports; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-20 10:19:23 -05:00
										 |  |  | Flags._constants = { | 
					
						
							|  |  |  | 	states: ['open', 'wip', 'resolved', 'rejected'], | 
					
						
							|  |  |  | 	state_class: { | 
					
						
							|  |  |  | 		open: 'info', | 
					
						
							|  |  |  | 		wip: 'warning', | 
					
						
							|  |  |  | 		resolved: 'success', | 
					
						
							|  |  |  | 		rejected: 'danger', | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | Flags.init = async function () { | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 	// Query plugins for custom filter strategies and merge into core filter strategies
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	function prepareSets(sets, orSets, prefix, value) { | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 		if (!Array.isArray(value)) { | 
					
						
							|  |  |  | 			sets.push(prefix + value); | 
					
						
							|  |  |  | 		} else if (value.length) { | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 			value.forEach(x => orSets.push(prefix + x)); | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	const hookData = { | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 		filters: { | 
					
						
							|  |  |  | 			type: function (sets, orSets, key) { | 
					
						
							|  |  |  | 				prepareSets(sets, orSets, 'flags:byType:', key); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			state: function (sets, orSets, key) { | 
					
						
							|  |  |  | 				prepareSets(sets, orSets, 'flags:byState:', key); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			reporterId: function (sets, orSets, key) { | 
					
						
							|  |  |  | 				prepareSets(sets, orSets, 'flags:byReporter:', key); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			assignee: function (sets, orSets, key) { | 
					
						
							|  |  |  | 				prepareSets(sets, orSets, 'flags:byAssignee:', key); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			targetUid: function (sets, orSets, key) { | 
					
						
							|  |  |  | 				prepareSets(sets, orSets, 'flags:byTargetUid:', key); | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			cid: function (sets, orSets, key) { | 
					
						
							|  |  |  | 				prepareSets(sets, orSets, 'flags:byCid:', key); | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2018-03-09 12:57:52 -05:00
										 |  |  | 			page: function () {	/* noop */ }, | 
					
						
							|  |  |  | 			perPage: function () {	/* noop */ }, | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 			quick: function (sets, orSets, key, uid) { | 
					
						
							|  |  |  | 				switch (key) { | 
					
						
							| 
									
										
										
										
											2020-06-03 11:25:25 -04:00
										 |  |  | 					case 'mine': | 
					
						
							|  |  |  | 						sets.push('flags:byAssignee:' + uid); | 
					
						
							|  |  |  | 						break; | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		helpers: { | 
					
						
							|  |  |  | 			prepareSets: prepareSets, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	try { | 
					
						
							|  |  |  | 		const data = await plugins.fireHook('filter:flags.getFilters', hookData); | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | 		Flags._filters = data.filters; | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	} catch (err) { | 
					
						
							| 
									
										
										
										
											2020-06-20 23:32:12 -04:00
										 |  |  | 		winston.error('[flags/init] Could not retrieve filters', err.stack); | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 		Flags._filters = {}; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-04 10:09:37 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | Flags.get = async function (flagId) { | 
					
						
							|  |  |  | 	const [base, history, notes] = await Promise.all([ | 
					
						
							|  |  |  | 		db.getObject('flag:' + flagId), | 
					
						
							|  |  |  | 		Flags.getHistory(flagId), | 
					
						
							|  |  |  | 		Flags.getNotes(flagId), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 	if (!base) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const [userObj, targetObj] = await Promise.all([ | 
					
						
							|  |  |  | 		user.getUserFields(base.uid, ['username', 'userslug', 'picture', 'reputation']), | 
					
						
							|  |  |  | 		Flags.getTarget(base.type, base.targetId, 0), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const flagObj = { | 
					
						
							|  |  |  | 		state: 'open', | 
					
						
							| 
									
										
										
										
											2020-04-23 21:50:08 -04:00
										 |  |  | 		assignee: null, | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 		...base, | 
					
						
							|  |  |  | 		description: validator.escape(base.description), | 
					
						
							|  |  |  | 		datetimeISO: utils.toISOString(base.datetime), | 
					
						
							|  |  |  | 		target_readable: base.type.charAt(0).toUpperCase() + base.type.slice(1) + ' ' + base.targetId, | 
					
						
							|  |  |  | 		target: targetObj, | 
					
						
							|  |  |  | 		history: history, | 
					
						
							|  |  |  | 		notes: notes, | 
					
						
							|  |  |  | 		reporter: userObj, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	const data = await plugins.fireHook('filter:flags.get', { | 
					
						
							|  |  |  | 		flag: flagObj, | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 	return data.flag; | 
					
						
							| 
									
										
										
										
											2016-11-25 14:17:51 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | Flags.list = async function (filters, uid) { | 
					
						
							|  |  |  | 	filters = filters || {}; | 
					
						
							| 
									
										
										
										
											2016-11-25 12:43:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	let sets = []; | 
					
						
							|  |  |  | 	const orSets = []; | 
					
						
							| 
									
										
										
										
											2017-01-03 13:38:06 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-09 12:57:52 -05:00
										 |  |  | 	// Default filter
 | 
					
						
							|  |  |  | 	filters.page = filters.hasOwnProperty('page') ? Math.abs(parseInt(filters.page, 10) || 1) : 1; | 
					
						
							|  |  |  | 	filters.perPage = filters.hasOwnProperty('perPage') ? Math.abs(parseInt(filters.perPage, 10) || 20) : 20; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (var type in filters) { | 
					
						
							|  |  |  | 		if (filters.hasOwnProperty(type)) { | 
					
						
							|  |  |  | 			if (Flags._filters.hasOwnProperty(type)) { | 
					
						
							|  |  |  | 				Flags._filters[type](sets, orSets, filters[type], uid); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				winston.warn('[flags/list] No flag filter type found: ' + type); | 
					
						
							| 
									
										
										
										
											2016-12-02 15:28:28 -05:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-03 13:38:06 -05:00
										 |  |  | 	sets = (sets.length || orSets.length) ? sets : ['flags:datetime'];	// No filter default
 | 
					
						
							| 
									
										
										
										
											2016-12-02 15:28:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	let flagIds = []; | 
					
						
							|  |  |  | 	if (sets.length === 1) { | 
					
						
							|  |  |  | 		flagIds = await db.getSortedSetRevRange(sets[0], 0, -1); | 
					
						
							|  |  |  | 	} else if (sets.length > 1) { | 
					
						
							|  |  |  | 		flagIds = await db.getSortedSetRevIntersect({ sets: sets, start: 0, stop: -1, aggregate: 'MAX' }); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	if (orSets.length) { | 
					
						
							|  |  |  | 		const _flagIds = await db.getSortedSetRevUnion({ sets: orSets, start: 0, stop: -1, aggregate: 'MAX' }); | 
					
						
							|  |  |  | 		if (sets.length) { | 
					
						
							|  |  |  | 			// If flag ids are already present, return a subset of flags that are in both sets
 | 
					
						
							|  |  |  | 			flagIds = _.intersection(flagIds, _flagIds); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			// Otherwise, return all flags returned via orSets
 | 
					
						
							|  |  |  | 			flagIds = _.union(flagIds, _flagIds); | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	// Create subset for parsing based on page number (n=20)
 | 
					
						
							|  |  |  | 	const flagsPerPage = Math.abs(parseInt(filters.perPage, 10) || 1); | 
					
						
							|  |  |  | 	const pageCount = Math.ceil(flagIds.length / flagsPerPage); | 
					
						
							|  |  |  | 	flagIds = flagIds.slice((filters.page - 1) * flagsPerPage, filters.page * flagsPerPage); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const flags = await Promise.all(flagIds.map(async (flagId) => { | 
					
						
							|  |  |  | 		let flagObj = await db.getObject('flag:' + flagId); | 
					
						
							|  |  |  | 		const userObj = await user.getUserFields(flagObj.uid, ['username', 'picture']); | 
					
						
							|  |  |  | 		flagObj = { | 
					
						
							|  |  |  | 			state: 'open', | 
					
						
							| 
									
										
										
										
											2020-04-23 21:50:08 -04:00
										 |  |  | 			assignee: null, | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 			...flagObj, | 
					
						
							|  |  |  | 			reporter: { | 
					
						
							|  |  |  | 				username: userObj.username, | 
					
						
							|  |  |  | 				picture: userObj.picture, | 
					
						
							|  |  |  | 				'icon:bgColor': userObj['icon:bgColor'], | 
					
						
							|  |  |  | 				'icon:text': userObj['icon:text'], | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2020-01-20 10:19:23 -05:00
										 |  |  | 		flagObj.labelClass = Flags._constants.state_class[flagObj.state]; | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return Object.assign(flagObj, { | 
					
						
							|  |  |  | 			description: validator.escape(String(flagObj.description)), | 
					
						
							|  |  |  | 			target_readable: flagObj.type.charAt(0).toUpperCase() + flagObj.type.slice(1) + ' ' + flagObj.targetId, | 
					
						
							|  |  |  | 			datetimeISO: utils.toISOString(flagObj.datetime), | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	})); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const data = await plugins.fireHook('filter:flags.list', { | 
					
						
							|  |  |  | 		flags: flags, | 
					
						
							|  |  |  | 		page: filters.page, | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2016-12-07 12:07:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	return { | 
					
						
							|  |  |  | 		flags: data.flags, | 
					
						
							|  |  |  | 		page: data.page, | 
					
						
							|  |  |  | 		pageCount: pageCount, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2017-02-24 12:47:46 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | Flags.validate = async function (payload) { | 
					
						
							|  |  |  | 	const [target, reporter] = await Promise.all([ | 
					
						
							|  |  |  | 		Flags.getTarget(payload.type, payload.id, payload.uid), | 
					
						
							|  |  |  | 		user.getUserData(payload.uid), | 
					
						
							|  |  |  | 	]); | 
					
						
							| 
									
										
										
										
											2016-12-07 12:07:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	if (!target) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | 	} else if (target.deleted) { | 
					
						
							|  |  |  | 		throw new Error('[[error:post-deleted]]'); | 
					
						
							|  |  |  | 	} else if (!reporter || !reporter.userslug) { | 
					
						
							|  |  |  | 		throw new Error('[[error:no-user]]'); | 
					
						
							|  |  |  | 	} else if (reporter.banned) { | 
					
						
							|  |  |  | 		throw new Error('[[error:user-banned]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-24 12:47:46 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	if (payload.type === 'post') { | 
					
						
							|  |  |  | 		const editable = await privileges.posts.canEdit(payload.id, payload.uid); | 
					
						
							|  |  |  | 		if (!editable.flag && !meta.config['reputation:disabled'] && reporter.reputation < meta.config['min:rep:flag']) { | 
					
						
							|  |  |  | 			throw new Error('[[error:not-enough-reputation-to-flag]]'); | 
					
						
							| 
									
										
										
										
											2017-02-24 12:47:46 -05:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	} else if (payload.type === 'user') { | 
					
						
							|  |  |  | 		const editable = await privileges.users.canEdit(payload.uid, payload.id); | 
					
						
							|  |  |  | 		if (!editable && !meta.config['reputation:disabled'] && reporter.reputation < meta.config['min:rep:flag']) { | 
					
						
							|  |  |  | 			throw new Error('[[error:not-enough-reputation-to-flag]]'); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | Flags.getNotes = async function (flagId) { | 
					
						
							|  |  |  | 	let notes = await db.getSortedSetRevRangeWithScores('flag:' + flagId + ':notes', 0, -1); | 
					
						
							| 
									
										
										
										
											2020-07-13 20:29:05 -04:00
										 |  |  | 	notes = await modifyNotes(notes); | 
					
						
							|  |  |  | 	return notes; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Flags.getNote = async function (flagId, datetime) { | 
					
						
							|  |  |  | 	let notes = await db.getSortedSetRangeByScoreWithScores('flag:' + flagId + ':notes', 0, 1, datetime, datetime); | 
					
						
							|  |  |  | 	if (!notes.length) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	notes = await modifyNotes(notes); | 
					
						
							|  |  |  | 	return notes[0]; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function modifyNotes(notes) { | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	const uids = []; | 
					
						
							|  |  |  | 	notes = notes.map(function (note) { | 
					
						
							|  |  |  | 		const noteObj = JSON.parse(note.value); | 
					
						
							|  |  |  | 		uids.push(noteObj[0]); | 
					
						
							|  |  |  | 		return { | 
					
						
							|  |  |  | 			uid: noteObj[0], | 
					
						
							|  |  |  | 			content: noteObj[1], | 
					
						
							|  |  |  | 			datetime: note.score, | 
					
						
							|  |  |  | 			datetimeISO: utils.toISOString(note.score), | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 	const userData = await user.getUsersFields(uids, ['username', 'userslug', 'picture']); | 
					
						
							|  |  |  | 	return notes.map(function (note, idx) { | 
					
						
							|  |  |  | 		note.user = userData[idx]; | 
					
						
							|  |  |  | 		note.content = validator.escape(note.content); | 
					
						
							|  |  |  | 		return note; | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2020-07-13 20:29:05 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Flags.deleteNote = async function (flagId, datetime) { | 
					
						
							|  |  |  | 	const note = await db.getSortedSetRangeByScore('flag:' + flagId + ':notes', 0, 1, datetime, datetime); | 
					
						
							|  |  |  | 	if (!note.length) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	await db.sortedSetRemove('flag:' + flagId + ':notes', note[0]); | 
					
						
							| 
									
										
										
										
											2016-11-30 20:34:06 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | Flags.create = async function (type, id, uid, reason, timestamp) { | 
					
						
							|  |  |  | 	let doHistoryAppend = false; | 
					
						
							|  |  |  | 	if (!timestamp) { | 
					
						
							| 
									
										
										
										
											2016-12-09 14:33:59 -05:00
										 |  |  | 		timestamp = Date.now(); | 
					
						
							|  |  |  | 		doHistoryAppend = true; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-01-22 12:14:50 -05:00
										 |  |  | 	const [flagExists, targetExists, canFlag, targetUid, targetCid] = await Promise.all([ | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 		// Sanity checks
 | 
					
						
							|  |  |  | 		Flags.exists(type, id, uid), | 
					
						
							|  |  |  | 		Flags.targetExists(type, id), | 
					
						
							| 
									
										
										
										
											2020-01-22 12:14:50 -05:00
										 |  |  | 		Flags.canFlag(type, id, uid), | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 		// Extra data for zset insertion
 | 
					
						
							|  |  |  | 		Flags.getTargetUid(type, id), | 
					
						
							|  |  |  | 		Flags.getTargetCid(type, id), | 
					
						
							|  |  |  | 	]); | 
					
						
							| 
									
										
										
										
											2020-01-22 12:14:50 -05:00
										 |  |  | 	if (flagExists) { | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 		throw new Error('[[error:already-flagged]]'); | 
					
						
							|  |  |  | 	} else if (!targetExists) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							| 
									
										
										
										
											2020-01-22 12:14:50 -05:00
										 |  |  | 	} else if (!canFlag) { | 
					
						
							|  |  |  | 		throw new Error('[[error:no-privileges]]'); | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	const flagId = await db.incrObjectField('global', 'nextFlagId'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	await db.setObject('flag:' + flagId, { | 
					
						
							|  |  |  | 		flagId: flagId, | 
					
						
							|  |  |  | 		type: type, | 
					
						
							|  |  |  | 		targetId: id, | 
					
						
							|  |  |  | 		description: reason, | 
					
						
							|  |  |  | 		uid: uid, | 
					
						
							|  |  |  | 		datetime: timestamp, | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 	await db.sortedSetAdd('flags:datetime', timestamp, flagId); // by time, the default
 | 
					
						
							|  |  |  | 	await db.sortedSetAdd('flags:byReporter:' + uid, timestamp, flagId); // by reporter
 | 
					
						
							|  |  |  | 	await db.sortedSetAdd('flags:byType:' + type, timestamp, flagId);	// by flag type
 | 
					
						
							|  |  |  | 	await db.sortedSetAdd('flags:hash', flagId, [type, id, uid].join(':')); // save zset for duplicate checking
 | 
					
						
							|  |  |  | 	await analytics.increment('flags'); // some fancy analytics
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (targetUid) { | 
					
						
							|  |  |  | 		await db.sortedSetAdd('flags:byTargetUid:' + targetUid, timestamp, flagId); // by target uid
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	if (targetCid) { | 
					
						
							|  |  |  | 		await db.sortedSetAdd('flags:byCid:' + targetCid, timestamp, flagId); // by target cid
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-11-12 08:45:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	if (type === 'post') { | 
					
						
							|  |  |  | 		await db.sortedSetAdd('flags:byPid:' + id, timestamp, flagId);	// by target pid
 | 
					
						
							|  |  |  | 		if (targetUid) { | 
					
						
							| 
									
										
										
										
											2020-05-02 15:34:58 -04:00
										 |  |  | 			await user.incrementUserFlagsBy(targetUid, 1); | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-24 12:47:46 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	if (doHistoryAppend) { | 
					
						
							|  |  |  | 		await Flags.update(flagId, uid, { state: 'open' }); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-09 14:33:59 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-26 22:51:11 -04:00
										 |  |  | 	return await Flags.get(flagId); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | Flags.exists = async function (type, id, uid) { | 
					
						
							|  |  |  | 	return await db.isSortedSetMember('flags:hash', [type, id, uid].join(':')); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-22 12:14:50 -05:00
										 |  |  | Flags.canFlag = async function (type, id, uid) { | 
					
						
							|  |  |  | 	if (type === 'user') { | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (type === 'post') { | 
					
						
							|  |  |  | 		return await privileges.posts.can('topics:read', id, uid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	throw new Error('[[error:invalid-data]]'); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | Flags.getTarget = async function (type, id, uid) { | 
					
						
							|  |  |  | 	if (type === 'user') { | 
					
						
							|  |  |  | 		const userData = await user.getUserData(id); | 
					
						
							|  |  |  | 		return userData && userData.uid ? userData : {}; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (type === 'post') { | 
					
						
							|  |  |  | 		let postData = await posts.getPostData(id); | 
					
						
							|  |  |  | 		if (!postData) { | 
					
						
							|  |  |  | 			return {}; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		postData = await posts.parsePost(postData); | 
					
						
							|  |  |  | 		postData = await topics.addPostData([postData], uid); | 
					
						
							|  |  |  | 		return postData[0]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	throw new Error('[[error:invalid-data]]'); | 
					
						
							| 
									
										
										
										
											2016-12-19 11:16:03 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | Flags.targetExists = async function (type, id) { | 
					
						
							|  |  |  | 	if (type === 'post') { | 
					
						
							|  |  |  | 		return await posts.exists(id); | 
					
						
							|  |  |  | 	} else if (type === 'user') { | 
					
						
							|  |  |  | 		return await user.exists(id); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	throw new Error('[[error:invalid-data]]'); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | Flags.getTargetUid = async function (type, id) { | 
					
						
							|  |  |  | 	if (type === 'post') { | 
					
						
							|  |  |  | 		return await posts.getPostField(id, 'uid'); | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | 	return id; | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-24 00:30:59 -04:00
										 |  |  | Flags.getTargetCid = async function (type, id) { | 
					
						
							|  |  |  | 	if (type === 'post') { | 
					
						
							|  |  |  | 		return await posts.getCidByPid(id); | 
					
						
							| 
									
										
										
										
											2016-12-14 15:53:57 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-24 12:22:40 -04:00
										 |  |  | 	return null; | 
					
						
							| 
									
										
										
										
											2016-12-14 15:53:57 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | Flags.update = async function (flagId, uid, changeset) { | 
					
						
							| 
									
										
										
										
											2020-05-01 13:32:20 -04:00
										 |  |  | 	const current = await db.getObjectFields('flag:' + flagId, ['uid', 'state', 'assignee', 'type', 'targetId']); | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	const now = changeset.datetime || Date.now(); | 
					
						
							|  |  |  | 	const notifyAssignee = async function (assigneeId) { | 
					
						
							| 
									
										
										
										
											2018-09-18 17:28:24 -04:00
										 |  |  | 		if (assigneeId === '' || parseInt(uid, 10) === parseInt(assigneeId, 10)) { | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 			return; | 
					
						
							| 
									
										
										
										
											2017-03-17 13:16:16 -04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		const notifObj = await notifications.create({ | 
					
						
							| 
									
										
										
										
											2017-03-17 13:16:16 -04:00
										 |  |  | 			type: 'my-flags', | 
					
						
							|  |  |  | 			bodyShort: '[[notifications:flag_assigned_to_you, ' + flagId + ']]', | 
					
						
							|  |  |  | 			bodyLong: '', | 
					
						
							|  |  |  | 			path: '/flags/' + flagId, | 
					
						
							|  |  |  | 			nid: 'flags:assign:' + flagId + ':uid:' + assigneeId, | 
					
						
							|  |  |  | 			from: uid, | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		await notifications.push(notifObj, [assigneeId]); | 
					
						
							| 
									
										
										
										
											2017-03-17 13:16:16 -04:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2020-01-20 10:19:23 -05:00
										 |  |  | 	const isAssignable = async function (assigneeId) { | 
					
						
							|  |  |  | 		let allowed = false; | 
					
						
							|  |  |  | 		allowed = await user.isAdminOrGlobalMod(assigneeId); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Mods are also allowed to be assigned, if flag target is post in uid's moderated cid
 | 
					
						
							|  |  |  | 		if (!allowed && current.type === 'post') { | 
					
						
							|  |  |  | 			const cid = await posts.getCidByPid(current.targetId); | 
					
						
							|  |  |  | 			allowed = await user.isModerator(assigneeId, cid); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return allowed; | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-20 10:19:23 -05:00
										 |  |  | 	// Retrieve existing flag data to compare for history-saving/reference purposes
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	const tasks = []; | 
					
						
							|  |  |  | 	for (var prop in changeset) { | 
					
						
							|  |  |  | 		if (changeset.hasOwnProperty(prop)) { | 
					
						
							|  |  |  | 			if (current[prop] === changeset[prop]) { | 
					
						
							|  |  |  | 				delete changeset[prop]; | 
					
						
							|  |  |  | 			} else if (prop === 'state') { | 
					
						
							| 
									
										
										
										
											2020-01-20 10:19:23 -05:00
										 |  |  | 				if (!Flags._constants.states.includes(changeset[prop])) { | 
					
						
							|  |  |  | 					delete changeset[prop]; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					tasks.push(db.sortedSetAdd('flags:byState:' + changeset[prop], now, flagId)); | 
					
						
							|  |  |  | 					tasks.push(db.sortedSetRemove('flags:byState:' + current[prop], flagId)); | 
					
						
							| 
									
										
										
										
											2020-05-01 13:32:20 -04:00
										 |  |  | 					if (changeset[prop] === 'resolved' || changeset[prop] === 'rejected') { | 
					
						
							|  |  |  | 						tasks.push(notifications.rescind('flag:' + current.type + ':' + current.targetId + ':uid:' + current.uid)); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2020-01-20 10:19:23 -05:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 			} else if (prop === 'assignee') { | 
					
						
							| 
									
										
										
										
											2020-01-20 10:19:23 -05:00
										 |  |  | 				/* eslint-disable-next-line */ | 
					
						
							|  |  |  | 				if (!await isAssignable(parseInt(changeset[prop], 10))) { | 
					
						
							|  |  |  | 					delete changeset[prop]; | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					tasks.push(db.sortedSetAdd('flags:byAssignee:' + changeset[prop], now, flagId)); | 
					
						
							|  |  |  | 					tasks.push(notifyAssignee(changeset[prop])); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	if (!Object.keys(changeset).length) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-12 10:03:45 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	tasks.push(db.setObject('flag:' + flagId, changeset)); | 
					
						
							|  |  |  | 	tasks.push(Flags.appendHistory(flagId, uid, changeset)); | 
					
						
							|  |  |  | 	tasks.push(plugins.fireHook('action:flags.update', { flagId: flagId, changeset: changeset, uid: uid })); | 
					
						
							|  |  |  | 	await Promise.all(tasks); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | Flags.getHistory = async function (flagId) { | 
					
						
							|  |  |  | 	const uids = []; | 
					
						
							|  |  |  | 	let history = await db.getSortedSetRevRangeWithScores('flag:' + flagId + ':history', 0, -1); | 
					
						
							| 
									
										
										
										
											2020-05-26 20:27:16 -04:00
										 |  |  | 	const flagData = await db.getObjectFields('flag:' + flagId, ['type', 'targetId']); | 
					
						
							|  |  |  | 	const targetUid = await Flags.getTargetUid(flagData.type, flagData.targetId); | 
					
						
							| 
									
										
										
										
											2016-12-01 16:22:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	history = history.map(function (entry) { | 
					
						
							|  |  |  | 		entry.value = JSON.parse(entry.value); | 
					
						
							| 
									
										
										
										
											2016-12-01 16:22:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		uids.push(entry.value[0]); | 
					
						
							| 
									
										
										
										
											2016-12-02 12:34:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		// Deserialise changeset
 | 
					
						
							|  |  |  | 		const changeset = entry.value[1]; | 
					
						
							|  |  |  | 		if (changeset.hasOwnProperty('state')) { | 
					
						
							|  |  |  | 			changeset.state = changeset.state === undefined ? '' : '[[flags:state-' + changeset.state + ']]'; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-12-01 16:22:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		return { | 
					
						
							|  |  |  | 			uid: entry.value[0], | 
					
						
							|  |  |  | 			fields: changeset, | 
					
						
							|  |  |  | 			datetime: entry.score, | 
					
						
							|  |  |  | 			datetimeISO: utils.toISOString(entry.score), | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2020-05-26 20:27:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Append ban history and username change data
 | 
					
						
							| 
									
										
										
										
											2020-05-26 20:40:55 -04:00
										 |  |  | 	history = await mergeBanHistory(history, targetUid, uids); | 
					
						
							|  |  |  | 	history = await mergeUsernameEmailChanges(history, targetUid, uids); | 
					
						
							| 
									
										
										
										
											2020-05-26 20:27:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	const userData = await user.getUsersFields(uids, ['username', 'userslug', 'picture']); | 
					
						
							|  |  |  | 	history.forEach((event, idx) => { event.user = userData[idx]; }); | 
					
						
							| 
									
										
										
										
											2020-05-26 20:27:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Resort by date
 | 
					
						
							|  |  |  | 	history = history.sort((a, b) => b.datetime - a.datetime); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	return history; | 
					
						
							| 
									
										
										
										
											2016-12-01 16:22:10 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | Flags.appendHistory = async function (flagId, uid, changeset) { | 
					
						
							|  |  |  | 	const datetime = changeset.datetime || Date.now(); | 
					
						
							| 
									
										
										
										
											2016-12-09 14:33:59 -05:00
										 |  |  | 	delete changeset.datetime; | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	const payload = JSON.stringify([uid, changeset, datetime]); | 
					
						
							|  |  |  | 	await db.sortedSetAdd('flag:' + flagId + ':history', datetime, payload); | 
					
						
							| 
									
										
										
										
											2016-12-01 11:42:06 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | Flags.appendNote = async function (flagId, uid, note, datetime) { | 
					
						
							| 
									
										
										
										
											2020-07-14 13:38:23 -04:00
										 |  |  | 	if (datetime) { | 
					
						
							|  |  |  | 		await Flags.deleteNote(flagId, datetime); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	datetime = datetime || Date.now(); | 
					
						
							| 
									
										
										
										
											2020-07-14 13:38:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	const payload = JSON.stringify([uid, note]); | 
					
						
							|  |  |  | 	await db.sortedSetAdd('flag:' + flagId + ':notes', datetime, payload); | 
					
						
							| 
									
										
										
										
											2019-09-28 15:08:51 -04:00
										 |  |  | 	await Flags.appendHistory(flagId, uid, { | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		notes: null, | 
					
						
							|  |  |  | 		datetime: datetime, | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2016-11-23 19:41:35 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | Flags.notify = async function (flagObj, uid) { | 
					
						
							|  |  |  | 	const [admins, globalMods] = await Promise.all([ | 
					
						
							|  |  |  | 		groups.getMembers('administrators', 0, -1), | 
					
						
							|  |  |  | 		groups.getMembers('Global Moderators', 0, -1), | 
					
						
							|  |  |  | 	]); | 
					
						
							|  |  |  | 	let uids = admins.concat(globalMods); | 
					
						
							|  |  |  | 	let notifObj = null; | 
					
						
							|  |  |  | 	if (flagObj.type === 'post') { | 
					
						
							|  |  |  | 		const [title, cid] = await Promise.all([ | 
					
						
							|  |  |  | 			topics.getTitleByPid(flagObj.targetId), | 
					
						
							|  |  |  | 			posts.getCidByPid(flagObj.targetId), | 
					
						
							|  |  |  | 		]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const modUids = await categories.getModeratorUids([cid]); | 
					
						
							|  |  |  | 		const titleEscaped = utils.decodeHTMLEntities(title).replace(/%/g, '%').replace(/,/g, ','); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		notifObj = await notifications.create({ | 
					
						
							|  |  |  | 			type: 'new-post-flag', | 
					
						
							|  |  |  | 			bodyShort: '[[notifications:user_flagged_post_in, ' + flagObj.reporter.username + ', ' + titleEscaped + ']]', | 
					
						
							|  |  |  | 			bodyLong: flagObj.description, | 
					
						
							|  |  |  | 			pid: flagObj.targetId, | 
					
						
							|  |  |  | 			path: '/flags/' + flagObj.flagId, | 
					
						
							|  |  |  | 			nid: 'flag:post:' + flagObj.targetId + ':uid:' + uid, | 
					
						
							|  |  |  | 			from: uid, | 
					
						
							|  |  |  | 			mergeId: 'notifications:user_flagged_post_in|' + flagObj.targetId, | 
					
						
							|  |  |  | 			topicTitle: title, | 
					
						
							| 
									
										
										
										
											2017-02-24 12:47:46 -05:00
										 |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 		uids = uids.concat(modUids[0]); | 
					
						
							|  |  |  | 	} else if (flagObj.type === 'user') { | 
					
						
							|  |  |  | 		notifObj = await notifications.create({ | 
					
						
							|  |  |  | 			type: 'new-user-flag', | 
					
						
							|  |  |  | 			bodyShort: '[[notifications:user_flagged_user, ' + flagObj.reporter.username + ', ' + flagObj.target.username + ']]', | 
					
						
							|  |  |  | 			bodyLong: flagObj.description, | 
					
						
							|  |  |  | 			path: '/flags/' + flagObj.flagId, | 
					
						
							|  |  |  | 			nid: 'flag:user:' + flagObj.targetId + ':uid:' + uid, | 
					
						
							|  |  |  | 			from: uid, | 
					
						
							|  |  |  | 			mergeId: 'notifications:user_flagged_user|' + flagObj.targetId, | 
					
						
							| 
									
										
										
										
											2017-02-24 12:47:46 -05:00
										 |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-data]]'); | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-09-28 14:37:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	plugins.fireHook('action:flags.create', { | 
					
						
							|  |  |  | 		flag: flagObj, | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 	uids = uids.filter(_uid => parseInt(_uid, 10) !== parseInt(uid, 10)); | 
					
						
							|  |  |  | 	await notifications.push(notifObj, uids); | 
					
						
							| 
									
										
										
										
											2016-12-06 16:11:56 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-07-22 18:16:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-26 20:40:55 -04:00
										 |  |  | async function mergeBanHistory(history, targetUid, uids) { | 
					
						
							|  |  |  | 	let recentBans = await db.getSortedSetRevRange('uid:' + targetUid + ':bans:timestamp', 0, 19); | 
					
						
							|  |  |  | 	recentBans = await db.getObjects(recentBans); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return history.concat(recentBans.reduce((memo, cur) => { | 
					
						
							|  |  |  | 		uids.push(cur.fromUid); | 
					
						
							|  |  |  | 		memo.push({ | 
					
						
							|  |  |  | 			uid: cur.fromUid, | 
					
						
							|  |  |  | 			meta: [ | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					key: '[[user:banned]]', | 
					
						
							|  |  |  | 					value: cur.reason, | 
					
						
							|  |  |  | 					labelClass: 'danger', | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					key: '[[user:info.banned-expiry]]', | 
					
						
							|  |  |  | 					value: new Date(parseInt(cur.expire, 10)).toISOString(), | 
					
						
							|  |  |  | 					labelClass: 'default', | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			], | 
					
						
							|  |  |  | 			datetime: parseInt(cur.timestamp, 10), | 
					
						
							|  |  |  | 			datetimeISO: utils.toISOString(parseInt(cur.timestamp, 10)), | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return memo; | 
					
						
							|  |  |  | 	}, [])); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function mergeUsernameEmailChanges(history, targetUid, uids) { | 
					
						
							|  |  |  | 	const usernameChanges = await user.getHistory('user:' + targetUid + ':usernames'); | 
					
						
							|  |  |  | 	const emailChanges = await user.getHistory('user:' + targetUid + ':emails'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return history.concat(usernameChanges.reduce((memo, changeObj) => { | 
					
						
							|  |  |  | 		uids.push(targetUid); | 
					
						
							|  |  |  | 		memo.push({ | 
					
						
							|  |  |  | 			uid: targetUid, | 
					
						
							|  |  |  | 			meta: [ | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					key: '[[user:change_username]]', | 
					
						
							|  |  |  | 					value: changeObj.value, | 
					
						
							|  |  |  | 					labelClass: 'primary', | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			], | 
					
						
							|  |  |  | 			datetime: changeObj.timestamp, | 
					
						
							|  |  |  | 			datetimeISO: changeObj.timestampISO, | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return memo; | 
					
						
							|  |  |  | 	}, [])).concat(emailChanges.reduce((memo, changeObj) => { | 
					
						
							|  |  |  | 		uids.push(targetUid); | 
					
						
							|  |  |  | 		memo.push({ | 
					
						
							|  |  |  | 			uid: targetUid, | 
					
						
							|  |  |  | 			meta: [ | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					key: '[[user:change_email]]', | 
					
						
							|  |  |  | 					value: changeObj.value, | 
					
						
							|  |  |  | 					labelClass: 'primary', | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			], | 
					
						
							|  |  |  | 			datetime: changeObj.timestamp, | 
					
						
							|  |  |  | 			datetimeISO: changeObj.timestampISO, | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return memo; | 
					
						
							|  |  |  | 	}, [])); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 18:16:18 -04:00
										 |  |  | require('./promisify')(Flags); |