| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | const validator = require('validator'); | 
					
						
							|  |  |  | const winston = require('winston'); | 
					
						
							|  |  |  | const nconf = require('nconf'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const user = require('../../user'); | 
					
						
							|  |  |  | const groups = require('../../groups'); | 
					
						
							|  |  |  | const plugins = require('../../plugins'); | 
					
						
							|  |  |  | const meta = require('../../meta'); | 
					
						
							|  |  |  | const utils = require('../../utils'); | 
					
						
							|  |  |  | const privileges = require('../../privileges'); | 
					
						
							| 
									
										
										
										
											2019-07-20 23:18:45 -04:00
										 |  |  | const translator = require('../../translator'); | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | const helpers = module.exports; | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | helpers.getUserDataByUserSlug = async function (userslug, callerUID) { | 
					
						
							|  |  |  | 	const uid = await user.getUidByUserslug(userslug); | 
					
						
							|  |  |  | 	if (!uid) { | 
					
						
							|  |  |  | 		return null; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-10-26 15:20:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	const results = await getAllData(uid, callerUID); | 
					
						
							|  |  |  | 	if (!results.userData) { | 
					
						
							|  |  |  | 		throw new Error('[[error:invalid-uid]]'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	await parseAboutMe(results.userData); | 
					
						
							| 
									
										
										
										
											2018-10-26 15:20:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	const userData = results.userData; | 
					
						
							|  |  |  | 	const userSettings = results.userSettings; | 
					
						
							|  |  |  | 	const isAdmin = results.isAdmin; | 
					
						
							|  |  |  | 	const isGlobalModerator = results.isGlobalModerator; | 
					
						
							|  |  |  | 	const isModerator = results.isModerator; | 
					
						
							|  |  |  | 	const isSelf = parseInt(callerUID, 10) === parseInt(userData.uid, 10); | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	userData.age = Math.max(0, userData.birthday ? Math.floor((new Date().getTime() - new Date(userData.birthday).getTime()) / 31536000000) : 0); | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	userData.emailClass = 'hide'; | 
					
						
							| 
									
										
										
										
											2016-03-20 15:07:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	if (!isAdmin && !isGlobalModerator && !isSelf && (!userSettings.showemail || meta.config.hideEmail)) { | 
					
						
							|  |  |  | 		userData.email = ''; | 
					
						
							|  |  |  | 	} else if (!userSettings.showemail) { | 
					
						
							|  |  |  | 		userData.emailClass = ''; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	if (!isAdmin && !isGlobalModerator && !isSelf && (!userSettings.showfullname || meta.config.hideFullname)) { | 
					
						
							|  |  |  | 		userData.fullname = ''; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	if (isAdmin || isSelf || ((isGlobalModerator || isModerator) && !results.isTargetAdmin)) { | 
					
						
							|  |  |  | 		userData.ips = results.ips; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	if (!isAdmin && !isGlobalModerator && !isModerator) { | 
					
						
							|  |  |  | 		userData.moderationNote = undefined; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-30 18:42:19 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	userData.isBlocked = results.isBlocked; | 
					
						
							|  |  |  | 	if (isAdmin || isSelf) { | 
					
						
							|  |  |  | 		userData.blocksCount = parseInt(userData.blocksCount, 10) || 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-07-05 14:18:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	userData.yourid = callerUID; | 
					
						
							|  |  |  | 	userData.theirid = userData.uid; | 
					
						
							|  |  |  | 	userData.isTargetAdmin = results.isTargetAdmin; | 
					
						
							|  |  |  | 	userData.isAdmin = isAdmin; | 
					
						
							|  |  |  | 	userData.isGlobalModerator = isGlobalModerator; | 
					
						
							|  |  |  | 	userData.isModerator = isModerator; | 
					
						
							|  |  |  | 	userData.isAdminOrGlobalModerator = isAdmin || isGlobalModerator; | 
					
						
							|  |  |  | 	userData.isAdminOrGlobalModeratorOrModerator = isAdmin || isGlobalModerator || isModerator; | 
					
						
							|  |  |  | 	userData.isSelfOrAdminOrGlobalModerator = isSelf || isAdmin || isGlobalModerator; | 
					
						
							|  |  |  | 	userData.canEdit = results.canEdit; | 
					
						
							|  |  |  | 	userData.canBan = results.canBanUser; | 
					
						
							|  |  |  | 	userData.canChangePassword = isAdmin || (isSelf && !meta.config['password:disableEdit']); | 
					
						
							|  |  |  | 	userData.isSelf = isSelf; | 
					
						
							|  |  |  | 	userData.isFollowing = results.isFollowing; | 
					
						
							|  |  |  | 	userData.showHidden = isSelf || isAdmin || (isGlobalModerator && !results.isTargetAdmin); | 
					
						
							|  |  |  | 	userData.groups = Array.isArray(results.groups) && results.groups.length ? results.groups[0] : []; | 
					
						
							|  |  |  | 	userData.disableSignatures = meta.config.disableSignatures === 1; | 
					
						
							|  |  |  | 	userData['reputation:disabled'] = meta.config['reputation:disabled'] === 1; | 
					
						
							|  |  |  | 	userData['downvote:disabled'] = meta.config['downvote:disabled'] === 1; | 
					
						
							|  |  |  | 	userData['email:confirmed'] = !!userData['email:confirmed']; | 
					
						
							|  |  |  | 	userData.profile_links = filterLinks(results.profile_menu.links, { | 
					
						
							|  |  |  | 		self: isSelf, | 
					
						
							|  |  |  | 		other: !isSelf, | 
					
						
							|  |  |  | 		moderator: isModerator, | 
					
						
							|  |  |  | 		globalMod: isGlobalModerator, | 
					
						
							|  |  |  | 		admin: isAdmin, | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2016-09-14 15:56:35 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	userData.sso = results.sso.associations; | 
					
						
							|  |  |  | 	userData.banned = userData.banned === 1; | 
					
						
							|  |  |  | 	userData.website = validator.escape(String(userData.website || '')); | 
					
						
							|  |  |  | 	userData.websiteLink = !userData.website.startsWith('http') ? 'http://' + userData.website : userData.website; | 
					
						
							|  |  |  | 	userData.websiteName = userData.website.replace(validator.escape('http://'), '').replace(validator.escape('https://'), ''); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	userData.fullname = validator.escape(String(userData.fullname || '')); | 
					
						
							|  |  |  | 	userData.location = validator.escape(String(userData.location || '')); | 
					
						
							|  |  |  | 	userData.signature = validator.escape(String(userData.signature || '')); | 
					
						
							|  |  |  | 	userData.birthday = validator.escape(String(userData.birthday || '')); | 
					
						
							|  |  |  | 	userData.moderationNote = validator.escape(String(userData.moderationNote || '')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (userData['cover:url']) { | 
					
						
							|  |  |  | 		userData['cover:url'] = userData['cover:url'].startsWith('http') ? userData['cover:url'] : (nconf.get('relative_path') + userData['cover:url']); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		userData['cover:url'] = require('../../coverPhoto').getDefaultProfileCover(userData.uid); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	userData['cover:position'] = validator.escape(String(userData['cover:position'] || '50% 50%')); | 
					
						
							|  |  |  | 	userData['username:disableEdit'] = !userData.isAdmin && meta.config['username:disableEdit']; | 
					
						
							|  |  |  | 	userData['email:disableEdit'] = !userData.isAdmin && meta.config['email:disableEdit']; | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	return userData; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2017-11-08 16:32:16 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | async function getAllData(uid, callerUID) { | 
					
						
							|  |  |  | 	return await utils.promiseParallel({ | 
					
						
							|  |  |  | 		userData: user.getUserData(uid), | 
					
						
							|  |  |  | 		isTargetAdmin: user.isAdministrator(uid), | 
					
						
							|  |  |  | 		userSettings: user.getSettings(uid), | 
					
						
							|  |  |  | 		isAdmin: user.isAdministrator(callerUID), | 
					
						
							|  |  |  | 		isGlobalModerator: user.isGlobalModerator(callerUID), | 
					
						
							|  |  |  | 		isModerator: user.isModeratorOfAnyCategory(callerUID), | 
					
						
							|  |  |  | 		isFollowing: user.isFollowing(callerUID, uid), | 
					
						
							|  |  |  | 		ips: user.getIPs(uid, 4), | 
					
						
							|  |  |  | 		profile_menu: getProfileMenu(uid, callerUID), | 
					
						
							|  |  |  | 		groups: groups.getUserGroups([uid]), | 
					
						
							|  |  |  | 		sso: plugins.fireHook('filter:auth.list', { uid: uid, associations: [] }), | 
					
						
							|  |  |  | 		canEdit: privileges.users.canEdit(callerUID, uid), | 
					
						
							|  |  |  | 		canBanUser: privileges.users.canBanUser(callerUID, uid), | 
					
						
							|  |  |  | 		isBlocked: user.blocks.is(uid, callerUID), | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-10-29 15:56:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | async function getProfileMenu(uid, callerUID) { | 
					
						
							|  |  |  | 	const links = [{ | 
					
						
							|  |  |  | 		id: 'info', | 
					
						
							|  |  |  | 		route: 'info', | 
					
						
							|  |  |  | 		name: '[[user:account_info]]', | 
					
						
							|  |  |  | 		visibility: { | 
					
						
							|  |  |  | 			self: false, | 
					
						
							|  |  |  | 			other: false, | 
					
						
							|  |  |  | 			moderator: true, | 
					
						
							|  |  |  | 			globalMod: true, | 
					
						
							|  |  |  | 			admin: true, | 
					
						
							| 
									
										
										
										
											2017-02-17 19:31:21 -07:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		id: 'sessions', | 
					
						
							|  |  |  | 		route: 'sessions', | 
					
						
							|  |  |  | 		name: '[[pages:account/sessions]]', | 
					
						
							|  |  |  | 		visibility: { | 
					
						
							|  |  |  | 			self: true, | 
					
						
							|  |  |  | 			other: false, | 
					
						
							|  |  |  | 			moderator: false, | 
					
						
							|  |  |  | 			globalMod: false, | 
					
						
							|  |  |  | 			admin: false, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (meta.config.gdpr_enabled) { | 
					
						
							|  |  |  | 		links.push({ | 
					
						
							|  |  |  | 			id: 'consent', | 
					
						
							|  |  |  | 			route: 'consent', | 
					
						
							|  |  |  | 			name: '[[user:consent.title]]', | 
					
						
							|  |  |  | 			visibility: { | 
					
						
							|  |  |  | 				self: true, | 
					
						
							|  |  |  | 				other: false, | 
					
						
							|  |  |  | 				moderator: false, | 
					
						
							|  |  |  | 				globalMod: false, | 
					
						
							|  |  |  | 				admin: false, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return await plugins.fireHook('filter:user.profileMenu', { | 
					
						
							|  |  |  | 		uid: uid, | 
					
						
							|  |  |  | 		callerUID: callerUID, | 
					
						
							|  |  |  | 		links: links, | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-09-25 01:52:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | async function parseAboutMe(userData) { | 
					
						
							| 
									
										
										
										
											2019-07-20 23:18:45 -04:00
										 |  |  | 	if (!userData.aboutme) { | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2019-07-20 23:18:45 -04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	userData.aboutme = validator.escape(String(userData.aboutme || '')); | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 	const parsed = await plugins.fireHook('filter:parse.aboutme', userData.aboutme); | 
					
						
							|  |  |  | 	userData.aboutmeParsed = translator.escape(parsed); | 
					
						
							| 
									
										
										
										
											2019-07-20 23:18:45 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-11 15:06:28 -05:00
										 |  |  | function filterLinks(links, states) { | 
					
						
							|  |  |  | 	return links.filter(function (link, index) { | 
					
						
							|  |  |  | 		// "public" is the old property, if visibility is defined, discard `public`
 | 
					
						
							|  |  |  | 		if (link.hasOwnProperty('public') && !link.hasOwnProperty('visibility')) { | 
					
						
							|  |  |  | 			winston.warn('[account/profileMenu (' + link.id + ')] Use of the `.public` property is deprecated, use `visibility` now'); | 
					
						
							|  |  |  | 			return link && (link.public || states.self); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Default visibility
 | 
					
						
							| 
									
										
										
										
											2019-08-13 14:33:37 -04:00
										 |  |  | 		link.visibility = { self: true, | 
					
						
							| 
									
										
										
										
											2017-01-11 15:06:28 -05:00
										 |  |  | 			other: true, | 
					
						
							|  |  |  | 			moderator: true, | 
					
						
							|  |  |  | 			globalMod: true, | 
					
						
							| 
									
										
										
										
											2017-02-24 12:47:46 -05:00
										 |  |  | 			admin: true, | 
					
						
							| 
									
										
										
										
											2019-08-13 14:33:37 -04:00
										 |  |  | 			...link.visibility }; | 
					
						
							| 
									
										
										
										
											2017-01-11 15:06:28 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		var permit = Object.keys(states).some(function (state) { | 
					
						
							| 
									
										
										
										
											2017-08-18 20:08:19 -04:00
										 |  |  | 			return states[state] && link.visibility[state]; | 
					
						
							| 
									
										
										
										
											2017-01-11 15:06:28 -05:00
										 |  |  | 		}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		links[index].public = permit; | 
					
						
							|  |  |  | 		return permit; | 
					
						
							| 
									
										
										
										
											2015-11-05 21:03:01 -05:00
										 |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-07-22 19:41:18 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | require('../../promisify')(helpers); |