feat: #7743, user/create, user/data, user/delete

v1.18.x
Barış Soner Uşaklı 6 years ago
parent c1660a1ace
commit d6e36c3166

@ -1,6 +1,5 @@
'use strict'; 'use strict';
var async = require('async');
var zxcvbn = require('zxcvbn'); var zxcvbn = require('zxcvbn');
var db = require('../database'); var db = require('../database');
var utils = require('../utils'); var utils = require('../utils');
@ -10,22 +9,17 @@ var meta = require('../meta');
module.exports = function (User) { module.exports = function (User) {
User.create = function (data, callback) { User.create = async function (data) {
data.username = data.username.trim(); data.username = data.username.trim();
data.userslug = utils.slugify(data.username); data.userslug = utils.slugify(data.username);
if (data.email !== undefined) { if (data.email !== undefined) {
data.email = String(data.email).trim(); data.email = String(data.email).trim();
} }
var timestamp = data.timestamp || Date.now(); const timestamp = data.timestamp || Date.now();
var userData;
var userNameChanged = false;
async.waterfall([ await User.isDataValid(data);
function (next) {
User.isDataValid(data, next); let userData = {
},
function (next) {
userData = {
username: data.username, username: data.username,
userslug: data.userslug, userslug: data.userslug,
email: data.email || '', email: data.email || '',
@ -48,33 +42,22 @@ module.exports = function (User) {
gdpr_consent: data.gdpr_consent === true ? 1 : 0, gdpr_consent: data.gdpr_consent === true ? 1 : 0,
acceptTos: data.acceptTos === true ? 1 : 0, acceptTos: data.acceptTos === true ? 1 : 0,
}; };
const renamedUsername = await User.uniqueUsername(userData);
User.uniqueUsername(userData, next); const userNameChanged = !!renamedUsername;
},
function (renamedUsername, next) {
userNameChanged = !!renamedUsername;
if (userNameChanged) { if (userNameChanged) {
userData.username = renamedUsername; userData.username = renamedUsername;
userData.userslug = utils.slugify(renamedUsername); userData.userslug = utils.slugify(renamedUsername);
} }
plugins.fireHook('filter:user.create', { user: userData, data: data }, next);
}, const results = await plugins.fireHook('filter:user.create', { user: userData, data: data });
function (results, next) {
userData = results.user; userData = results.user;
db.incrObjectField('global', 'nextUid', next);
}, const uid = await db.incrObjectField('global', 'nextUid');
function (uid, next) {
userData.uid = uid; userData.uid = uid;
db.setObject('user:' + uid, userData, next);
}, await db.setObject('user:' + uid, userData);
function (next) {
async.parallel([ const bulkAdd = [
function (next) {
db.incrObjectField('global', 'userCount', next);
},
function (next) {
const bulk = [
['username:uid', userData.uid, userData.username], ['username:uid', userData.uid, userData.username],
['user:' + userData.uid + ':usernames', timestamp, userData.username], ['user:' + userData.uid + ':usernames', timestamp, userData.username],
['username:sorted', 0, userData.username.toLowerCase() + ':' + userData.uid], ['username:sorted', 0, userData.username.toLowerCase() + ':' + userData.uid],
@ -86,139 +69,102 @@ module.exports = function (User) {
]; ];
if (parseInt(userData.uid, 10) !== 1) { if (parseInt(userData.uid, 10) !== 1) {
bulk.push(['users:notvalidated', timestamp, userData.uid]); bulkAdd.push(['users:notvalidated', timestamp, userData.uid]);
} }
if (userData.email) { if (userData.email) {
bulk.push(['email:uid', userData.uid, userData.email.toLowerCase()]); bulkAdd.push(['email:uid', userData.uid, userData.email.toLowerCase()]);
bulk.push(['email:sorted', 0, userData.email.toLowerCase() + ':' + userData.uid]); bulkAdd.push(['email:sorted', 0, userData.email.toLowerCase() + ':' + userData.uid]);
bulk.push(['user:' + userData.uid + ':emails', timestamp, userData.email]); bulkAdd.push(['user:' + userData.uid + ':emails', timestamp, userData.email]);
} }
db.sortedSetAddBulk(bulk, next);
}, await Promise.all([
function (next) { db.incrObjectField('global', 'userCount'),
groups.join('registered-users', userData.uid, next); db.sortedSetAddBulk(bulkAdd),
}, groups.join('registered-users', userData.uid),
function (next) { User.notifications.sendWelcomeNotification(userData.uid),
User.notifications.sendWelcomeNotification(userData.uid, next); storePassword(userData.uid, data.password),
}, User.updateDigestSetting(userData.uid, meta.config.dailyDigestFreq),
function (next) { ]);
if (userData.email && userData.uid > 1 && meta.config.requireEmailConfirmation) { if (userData.email && userData.uid > 1 && meta.config.requireEmailConfirmation) {
User.email.sendValidationEmail(userData.uid, { User.email.sendValidationEmail(userData.uid, {
email: userData.email, email: userData.email,
}); });
} }
if (userNameChanged) {
await User.notifications.sendNameChangeNotification(userData.uid, userData.username);
}
plugins.fireHook('action:user.create', { user: userData, data: data });
return userData.uid;
};
if (!data.password) { async function storePassword(uid, password) {
return next(); if (!password) {
return;
}
const hash = await User.hashPassword(password);
await Promise.all([
User.setUserField(uid, 'password', hash),
User.reset.updateExpiry(uid),
]);
} }
User.hashPassword(data.password, function (err, hash) { User.isDataValid = async function (userData) {
if (err) { if (userData.email && !utils.isEmailValid(userData.email)) {
return next(err); throw new Error('[[error:invalid-email]]');
} }
async.parallel([ if (!utils.isUserNameValid(userData.username) || !userData.userslug) {
async.apply(User.setUserField, userData.uid, 'password', hash), throw new Error('[[error:invalid-username, ' + userData.username + ']]');
async.apply(User.reset.updateExpiry, userData.uid),
], next);
});
},
function (next) {
User.updateDigestSetting(userData.uid, meta.config.dailyDigestFreq, next);
},
], next);
},
function (results, next) {
if (userNameChanged) {
User.notifications.sendNameChangeNotification(userData.uid, userData.username);
} }
plugins.fireHook('action:user.create', { user: userData });
next(null, userData.uid);
},
], callback);
};
User.isDataValid = function (userData, callback) {
async.parallel({
emailValid: function (next) {
if (userData.email) {
next(!utils.isEmailValid(userData.email) ? new Error('[[error:invalid-email]]') : null);
} else {
next();
}
},
userNameValid: function (next) {
next((!utils.isUserNameValid(userData.username) || !userData.userslug) ? new Error('[[error:invalid-username, ' + userData.username + ']]') : null);
},
passwordValid: function (next) {
if (userData.password) { if (userData.password) {
User.isPasswordValid(userData.password, next); await User.isPasswordValid(userData.password);
} else {
next();
} }
},
emailAvailable: function (next) {
if (userData.email) { if (userData.email) {
User.email.available(userData.email, function (err, available) { const available = await User.email.available(userData.email);
if (err) { if (!available) {
return next(err); throw new Error('[[error:email-taken]]');
} }
next(!available ? new Error('[[error:email-taken]]') : null);
});
} else {
next();
} }
},
}, function (err) {
callback(err);
});
}; };
User.isPasswordValid = function (password, minStrength, callback) { // this function doesnt need to be async, but there is exising code that uses it
if (typeof minStrength === 'function' && !callback) { // with a callback so it is marked async otherwise it breaks the callback code
callback = minStrength; User.isPasswordValid = async function (password, minStrength) {
minStrength = meta.config.minimumPasswordStrength; minStrength = minStrength || meta.config.minimumPasswordStrength;
}
// Sanity checks: Checks if defined and is string // Sanity checks: Checks if defined and is string
if (!password || !utils.isPasswordValid(password)) { if (!password || !utils.isPasswordValid(password)) {
return callback(new Error('[[error:invalid-password]]')); throw new Error('[[error:invalid-password]]');
} }
if (password.length < meta.config.minimumPasswordLength) { if (password.length < meta.config.minimumPasswordLength) {
return callback(new Error('[[reset_password:password_too_short]]')); throw new Error('[[reset_password:password_too_short]]');
} }
if (password.length > 512) { if (password.length > 512) {
return callback(new Error('[[error:password-too-long]]')); throw new Error('[[error:password-too-long]]');
} }
var strength = zxcvbn(password); var strength = zxcvbn(password);
if (strength.score < minStrength) { if (strength.score < minStrength) {
return callback(new Error('[[user:weak_password]]')); throw new Error('[[user:weak_password]]');
} }
callback();
}; };
User.uniqueUsername = function (userData, callback) { User.uniqueUsername = async function (userData) {
var numTries = 0; let numTries = 0;
function go(username) { let username = userData.username;
async.waterfall([ while (true) {
function (next) { /* eslint-disable no-await-in-loop */
meta.userOrGroupExists(username, next); const exists = await meta.userOrGroupExists(username);
},
function (exists) {
if (!exists) { if (!exists) {
return callback(null, numTries ? username : null); return numTries ? username : null;
} }
username = userData.username + ' ' + numTries.toString(32); username = userData.username + ' ' + numTries.toString(32);
numTries += 1; numTries += 1;
go(username);
},
], callback);
} }
go(userData.userslug);
}; };
}; };

@ -46,14 +46,33 @@ module.exports = function (User) {
'email:confirmed': 0, 'email:confirmed': 0,
}; };
User.getUsersFields = function (uids, fields, callback) { User.getUsersFields = async function (uids, fields) {
if (!Array.isArray(uids) || !uids.length) { if (!Array.isArray(uids) || !uids.length) {
return setImmediate(callback, null, []); return [];
} }
uids = uids.map(uid => (isNaN(uid) ? 0 : parseInt(uid, 10))); uids = uids.map(uid => (isNaN(uid) ? 0 : parseInt(uid, 10)));
var fieldsToRemove = []; const fieldsToRemove = [];
ensureRequiredFields(fields, fieldsToRemove);
const uniqueUids = _.uniq(uids).filter(uid => uid > 0);
const results = await plugins.fireHook('filter:user.whitelistFields', { uids: uids, whitelist: fieldWhitelist.slice() });
if (!fields.length) {
fields = results.whitelist;
} else {
// Never allow password retrieval via this method
fields = fields.filter(value => value !== 'password');
}
let users = await db.getObjectsFields(uniqueUids.map(uid => 'user:' + uid), fields);
users = uidsToUsers(uids, uniqueUids, users);
return await modifyUserData(users, fields, fieldsToRemove);
};
function ensureRequiredFields(fields, fieldsToRemove) {
function addField(field) { function addField(field) {
if (!fields.includes(field)) { if (!fields.includes(field)) {
fields.push(field); fields.push(field);
@ -76,60 +95,11 @@ module.exports = function (User) {
if (fields.includes('banned') && !fields.includes('banned:expire')) { if (fields.includes('banned') && !fields.includes('banned:expire')) {
addField('banned:expire'); addField('banned:expire');
} }
var uniqueUids = _.uniq(uids).filter(uid => uid > 0);
async.waterfall([
function (next) {
plugins.fireHook('filter:user.whitelistFields', { uids: uids, whitelist: fieldWhitelist.slice() }, next);
},
function (results, next) {
if (!fields.length) {
fields = results.whitelist;
} else {
// Never allow password retrieval via this method
fields = fields.filter(value => value !== 'password');
} }
db.getObjectsFields(uidsToUserKeys(uniqueUids), fields, next);
},
function (users, next) {
users = uidsToUsers(uids, uniqueUids, users);
modifyUserData(users, fields, fieldsToRemove, next);
},
], callback);
};
User.getUserField = function (uid, field, callback) {
User.getUserFields(uid, [field], function (err, user) {
callback(err, user ? user[field] : null);
});
};
User.getUserFields = function (uid, fields, callback) {
User.getUsersFields([uid], fields, function (err, users) {
callback(err, users ? users[0] : null);
});
};
User.getUserData = function (uid, callback) {
User.getUsersData([uid], function (err, users) {
callback(err, users ? users[0] : null);
});
};
User.getUsersData = function (uids, callback) {
User.getUsersFields(uids, [], callback);
};
function uidsToUsers(uids, uniqueUids, usersData) { function uidsToUsers(uids, uniqueUids, usersData) {
var uidToUser = uniqueUids.reduce(function (memo, uid, idx) { const uidToUser = _.zipObject(uniqueUids, usersData);
memo[uid] = usersData[idx]; const users = uids.map(function (uid) {
return memo;
}, {});
var users = uids.map(function (uid) {
const returnPayload = uidToUser[uid] || _.clone(User.guestData); const returnPayload = uidToUser[uid] || _.clone(User.guestData);
if (uid > 0 && !returnPayload.uid) { if (uid > 0 && !returnPayload.uid) {
returnPayload.oldUid = parseInt(uid, 10); returnPayload.oldUid = parseInt(uid, 10);
@ -140,14 +110,29 @@ module.exports = function (User) {
return users; return users;
} }
function uidsToUserKeys(uids) { User.getUserField = async function (uid, field) {
return uids.map(uid => 'user:' + uid); const user = await User.getUserFields(uid, [field]);
} return user ? user[field] : null;
};
User.getUserFields = async function (uid, fields) {
const users = await User.getUsersFields([uid], fields);
return users ? users[0] : null;
};
User.getUserData = async function (uid) {
const users = await User.getUsersData([uid]);
return users ? users[0] : null;
};
function modifyUserData(users, requestedFields, fieldsToRemove, callback) { User.getUsersData = async function (uids) {
async.map(users, function (user, next) { return await User.getUsersFields(uids, []);
};
async function modifyUserData(users, requestedFields, fieldsToRemove) {
users = await async.map(users, async function (user) {
if (!user) { if (!user) {
return next(null, user); return user;
} }
db.parseIntFields(user, intFields, requestedFields); db.parseIntFields(user, intFields, requestedFields);
@ -209,27 +194,18 @@ module.exports = function (User) {
} }
if (user.hasOwnProperty('banned') || user.hasOwnProperty('banned:expire')) { if (user.hasOwnProperty('banned') || user.hasOwnProperty('banned:expire')) {
var result = User.bans.calcExpiredFromUserData(user); const result = User.bans.calcExpiredFromUserData(user);
var unban = result.banned && result.banExpired; const unban = result.banned && result.banExpired;
user.banned_until = unban ? 0 : user['banned:expire']; user.banned_until = unban ? 0 : user['banned:expire'];
user.banned_until_readable = user.banned_until && !unban ? utils.toISOString(user.banned_until) : 'Not Banned'; user.banned_until_readable = user.banned_until && !unban ? utils.toISOString(user.banned_until) : 'Not Banned';
if (unban) { if (unban) {
return User.bans.unban(user.uid, function (err) { await User.bans.unban(user.uid);
if (err) {
return next(err);
}
user.banned = false; user.banned = false;
next(null, user);
});
}
} }
next(null, user);
}, function (err, users) {
if (err) {
return callback(err);
} }
plugins.fireHook('filter:users.get', users, callback); return user;
}); });
return await plugins.fireHook('filter:users.get', users);
} }
function parseGroupTitle(user) { function parseGroupTitle(user) {
@ -261,46 +237,30 @@ module.exports = function (User) {
return meta.config.defaultAvatar.startsWith('http') ? meta.config.defaultAvatar : nconf.get('relative_path') + meta.config.defaultAvatar; return meta.config.defaultAvatar.startsWith('http') ? meta.config.defaultAvatar : nconf.get('relative_path') + meta.config.defaultAvatar;
}; };
User.setUserField = function (uid, field, value, callback) { User.setUserField = async function (uid, field, value) {
User.setUserFields(uid, { [field]: value }, callback); await User.setUserFields(uid, { [field]: value });
}; };
User.setUserFields = function (uid, data, callback) { User.setUserFields = async function (uid, data) {
callback = callback || function () {}; await db.setObject('user:' + uid, data);
async.waterfall([
function (next) {
db.setObject('user:' + uid, data, next);
},
function (next) {
for (var field in data) { for (var field in data) {
if (data.hasOwnProperty(field)) { if (data.hasOwnProperty(field)) {
plugins.fireHook('action:user.set', { uid: uid, field: field, value: data[field], type: 'set' }); plugins.fireHook('action:user.set', { uid: uid, field: field, value: data[field], type: 'set' });
} }
} }
next();
},
], callback);
}; };
User.incrementUserFieldBy = function (uid, field, value, callback) { User.incrementUserFieldBy = async function (uid, field, value) {
incrDecrUserFieldBy(uid, field, value, 'increment', callback); return await incrDecrUserFieldBy(uid, field, value, 'increment');
}; };
User.decrementUserFieldBy = function (uid, field, value, callback) { User.decrementUserFieldBy = async function (uid, field, value) {
incrDecrUserFieldBy(uid, field, -value, 'decrement', callback); return await incrDecrUserFieldBy(uid, field, -value, 'decrement');
}; };
function incrDecrUserFieldBy(uid, field, value, type, callback) { async function incrDecrUserFieldBy(uid, field, value, type) {
callback = callback || function () {}; const newValue = await db.incrObjectFieldBy('user:' + uid, field, value);
async.waterfall([ plugins.fireHook('action:user.set', { uid: uid, field: field, value: newValue, type: type });
function (next) { return newValue;
db.incrObjectFieldBy('user:' + uid, field, value, next);
},
function (value, next) {
plugins.fireHook('action:user.set', { uid: uid, field: field, value: value, type: type });
next(null, value);
},
], callback);
} }
}; };

@ -17,66 +17,49 @@ var file = require('../file');
module.exports = function (User) { module.exports = function (User) {
var deletesInProgress = {}; var deletesInProgress = {};
User.delete = function (callerUid, uid, callback) { User.delete = async function (callerUid, uid) {
if (parseInt(uid, 10) <= 0) { if (parseInt(uid, 10) <= 0) {
return setImmediate(callback, new Error('[[error:invalid-uid]]')); throw new Error('[[error:invalid-uid]]');
} }
if (deletesInProgress[uid]) { if (deletesInProgress[uid]) {
return setImmediate(callback, new Error('[[error:already-deleting]]')); throw new Error('[[error:already-deleting]]');
} }
deletesInProgress[uid] = 'user.delete'; deletesInProgress[uid] = 'user.delete';
async.waterfall([ await removeFromSortedSets(uid);
function (next) { await deletePosts(callerUid, uid);
removeFromSortedSets(uid, next); await deleteTopics(callerUid, uid);
}, await deleteUploads(uid);
function (next) { const userData = await User.deleteAccount(uid);
deletePosts(callerUid, uid, next); return userData;
},
function (next) {
deleteTopics(callerUid, uid, next);
},
function (next) {
deleteUploads(uid, next);
},
function (next) {
User.deleteAccount(uid, next);
},
], callback);
}; };
function deletePosts(callerUid, uid, callback) { async function deletePosts(callerUid, uid) {
batch.processSortedSet('uid:' + uid + ':posts', function (ids, next) { await batch.processSortedSet('uid:' + uid + ':posts', async function (ids) {
async.eachSeries(ids, function (pid, next) { await async.eachSeries(ids, async function (pid) {
posts.purge(pid, callerUid, next); await posts.purge(pid, callerUid);
}, next); });
}, { alwaysStartAt: 0 }, callback); }, { alwaysStartAt: 0 });
} }
function deleteTopics(callerUid, uid, callback) { async function deleteTopics(callerUid, uid) {
batch.processSortedSet('uid:' + uid + ':topics', function (ids, next) { await batch.processSortedSet('uid:' + uid + ':topics', async function (ids) {
async.eachSeries(ids, function (tid, next) { await async.eachSeries(ids, async function (tid) {
topics.purge(tid, callerUid, next); await topics.purge(tid, callerUid);
}, next); });
}, { alwaysStartAt: 0 }, callback); }, { alwaysStartAt: 0 });
} }
function deleteUploads(uid, callback) { async function deleteUploads(uid) {
batch.processSortedSet('uid:' + uid + ':uploads', function (uploadNames, next) { await batch.processSortedSet('uid:' + uid + ':uploads', async function (uploadNames) {
async.waterfall([ await async.each(uploadNames, async function (uploadName) {
function (next) { await file.delete(path.join(nconf.get('upload_path'), uploadName));
async.each(uploadNames, function (uploadName, next) { });
file.delete(path.join(nconf.get('upload_path'), uploadName), next); await db.sortedSetRemove('uid:' + uid + ':uploads', uploadNames);
}, next); }, { alwaysStartAt: 0 });
},
function (next) {
db.sortedSetRemove('uid:' + uid + ':uploads', uploadNames, next);
},
], next);
}, { alwaysStartAt: 0 }, callback);
} }
function removeFromSortedSets(uid, callback) { async function removeFromSortedSets(uid) {
db.sortedSetsRemove([ await db.sortedSetsRemove([
'users:joindate', 'users:joindate',
'users:postcount', 'users:postcount',
'users:reputation', 'users:reputation',
@ -88,68 +71,29 @@ module.exports = function (User) {
'digest:day:uids', 'digest:day:uids',
'digest:week:uids', 'digest:week:uids',
'digest:month:uids', 'digest:month:uids',
], uid, callback); ], uid);
} }
User.deleteAccount = function (uid, callback) { User.deleteAccount = async function (uid) {
if (deletesInProgress[uid] === 'user.deleteAccount') { if (deletesInProgress[uid] === 'user.deleteAccount') {
return setImmediate(callback, new Error('[[error:already-deleting]]')); throw new Error('[[error:already-deleting]]');
} }
deletesInProgress[uid] = 'user.deleteAccount'; deletesInProgress[uid] = 'user.deleteAccount';
var userData;
async.waterfall([ await removeFromSortedSets(uid);
function (next) { const userData = await db.getObject('user:' + uid);
removeFromSortedSets(uid, next);
}, if (!userData || !userData.username) {
function (next) {
db.getObject('user:' + uid, next);
},
function (_userData, next) {
if (!_userData || !_userData.username) {
delete deletesInProgress[uid]; delete deletesInProgress[uid];
return callback(new Error('[[error:no-user]]')); throw new Error('[[error:no-user]]');
}
userData = _userData;
plugins.fireHook('static:user.delete', { uid: uid }, next);
},
function (next) {
deleteVotes(uid, next);
},
function (next) {
deleteChats(uid, next);
},
function (next) {
User.auth.revokeAllSessions(uid, next);
},
function (next) {
async.parallel([
function (next) {
db.sortedSetRemove('username:uid', userData.username, next);
},
function (next) {
db.sortedSetRemove('username:sorted', userData.username.toLowerCase() + ':' + uid, next);
},
function (next) {
db.sortedSetRemove('userslug:uid', userData.userslug, next);
},
function (next) {
db.sortedSetRemove('fullname:uid', userData.fullname, next);
},
function (next) {
if (userData.email) {
async.parallel([
async.apply(db.sortedSetRemove, 'email:uid', userData.email.toLowerCase()),
async.apply(db.sortedSetRemove, 'email:sorted', userData.email.toLowerCase() + ':' + uid),
], next);
} else {
next();
} }
},
function (next) { await plugins.fireHook('static:user.delete', { uid: uid });
db.decrObjectField('global', 'userCount', next); await deleteVotes(uid);
}, await deleteChats(uid);
function (next) { await User.auth.revokeAllSessions(uid);
var keys = [
const keys = [
'uid:' + uid + ':notifications:read', 'uid:' + uid + ':notifications:read',
'uid:' + uid + ':notifications:unread', 'uid:' + uid + ':notifications:unread',
'uid:' + uid + ':bookmarks', 'uid:' + uid + ':bookmarks',
@ -166,139 +110,87 @@ module.exports = function (User) {
'uid:' + uid + ':sessions', 'uid:' + uid + ':sessionUUID:sessionId', 'uid:' + uid + ':sessions', 'uid:' + uid + ':sessionUUID:sessionId',
'invitation:uid:' + uid, 'invitation:uid:' + uid,
]; ];
db.deleteAll(keys, next);
}, const bulkRemove = [
function (next) { ['username:uid', userData.username],
db.setRemove('invitation:uids', uid, next); ['username:sorted', userData.username.toLowerCase() + ':' + uid],
}, ['userslug:uid', userData.userslug],
function (next) { ['fullname:uid', userData.fullname],
deleteUserIps(uid, next); ];
}, if (userData.email) {
function (next) { bulkRemove.push(['email:uid', userData.email.toLowerCase()]);
deleteBans(uid, next); bulkRemove.push(['email:sorted', userData.email.toLowerCase() + ':' + uid]);
}, }
function (next) {
deleteUserFromFollowers(uid, next); await Promise.all([
}, db.sortedSetRemoveBulk(bulkRemove),
function (next) { db.decrObjectField('global', 'userCount'),
groups.leaveAllGroups(uid, next); db.deleteAll(keys),
}, db.setRemove('invitation:uids', uid),
], next); deleteUserIps(uid),
}, deleteBans(uid),
function (results, next) { deleteUserFromFollowers(uid),
db.deleteAll(['followers:' + uid, 'following:' + uid, 'user:' + uid], next); groups.leaveAllGroups(uid),
}, ]);
], function (err) { await db.deleteAll(['followers:' + uid, 'following:' + uid, 'user:' + uid]);
delete deletesInProgress[uid]; delete deletesInProgress[uid];
callback(err, userData); return userData;
});
}; };
function deleteVotes(uid, callback) { async function deleteVotes(uid) {
async.waterfall([ const [upvotedPids, downvotedPids] = await Promise.all([
function (next) { db.getSortedSetRange('uid:' + uid + ':upvote', 0, -1),
async.parallel({ db.getSortedSetRange('uid:' + uid + ':downvote', 0, -1),
upvotedPids: async.apply(db.getSortedSetRange, 'uid:' + uid + ':upvote', 0, -1), ]);
downvotedPids: async.apply(db.getSortedSetRange, 'uid:' + uid + ':downvote', 0, -1), const pids = _.uniq(upvotedPids.concat(downvotedPids).filter(Boolean));
}, next); await async.eachSeries(pids, async function (pid) {
}, await posts.unvote(pid, uid);
function (pids, next) {
pids = _.uniq(pids.upvotedPids.concat(pids.downvotedPids).filter(Boolean));
async.eachSeries(pids, function (pid, next) {
posts.unvote(pid, uid, next);
}, next);
},
], function (err) {
callback(err);
}); });
} }
function deleteChats(uid, callback) { async function deleteChats(uid) {
async.waterfall([ const roomIds = await db.getSortedSetRange('uid:' + uid + ':chat:rooms', 0, -1);
function (next) { const userKeys = roomIds.map(roomId => 'uid:' + uid + ':chat:room:' + roomId + ':mids');
db.getSortedSetRange('uid:' + uid + ':chat:rooms', 0, -1, next);
},
function (roomIds, next) {
var userKeys = roomIds.map(function (roomId) {
return 'uid:' + uid + ':chat:room:' + roomId + ':mids';
});
async.parallel([ await Promise.all([
async.apply(messaging.leaveRooms, uid, roomIds), messaging.leaveRooms(uid, roomIds),
async.apply(db.deleteAll, userKeys), db.deleteAll(userKeys),
], next); ]);
},
], function (err) {
callback(err);
});
} }
function deleteUserIps(uid, callback) { async function deleteUserIps(uid) {
async.waterfall([ const ips = await db.getSortedSetRange('uid:' + uid + ':ip', 0, -1);
function (next) { await db.sortedSetsRemove(ips.map(ip => 'ip:' + ip + ':uid'), uid);
db.getSortedSetRange('uid:' + uid + ':ip', 0, -1, next); await db.delete('uid:' + uid + ':ip');
},
function (ips, next) {
var keys = ips.map(function (ip) {
return 'ip:' + ip + ':uid';
});
db.sortedSetsRemove(keys, uid, next);
},
function (next) {
db.delete('uid:' + uid + ':ip', next);
},
], callback);
} }
function deleteBans(uid, callback) { async function deleteBans(uid) {
async.waterfall([ const bans = await db.getSortedSetRange('uid:' + uid + ':bans:timestamp', 0, -1);
function (next) { await db.deleteAll(bans);
db.getSortedSetRange('uid:' + uid + ':bans:timestamp', 0, -1, next); await db.delete('uid:' + uid + ':bans:timestamp');
},
function (bans, next) {
db.deleteAll(bans, next);
},
function (next) {
db.delete('uid:' + uid + ':bans:timestamp', next);
},
], callback);
} }
function deleteUserFromFollowers(uid, callback) { async function deleteUserFromFollowers(uid) {
async.parallel({ const [followers, following] = await Promise.all([
followers: async.apply(db.getSortedSetRange, 'followers:' + uid, 0, -1), db.getSortedSetRange('followers:' + uid, 0, -1),
following: async.apply(db.getSortedSetRange, 'following:' + uid, 0, -1), db.getSortedSetRange('following:' + uid, 0, -1),
}, function (err, results) { ]);
function updateCount(uids, name, fieldName, next) {
async.each(uids, function (uid, next) { async function updateCount(uids, name, fieldName) {
db.sortedSetCard(name + uid, function (err, count) { await async.each(uids, async function (uid) {
if (err) { let count = await db.sortedSetCard(name + uid);
return next(err);
}
count = parseInt(count, 10) || 0; count = parseInt(count, 10) || 0;
db.setObjectField('user:' + uid, fieldName, count, next); await db.setObjectField('user:' + uid, fieldName, count);
}); });
}, next);
} }
if (err) { const followingSets = followers.map(uid => 'following:' + uid);
return callback(err); const followerSets = following.map(uid => 'followers:' + uid);
}
var followingSets = results.followers.map(function (uid) {
return 'following:' + uid;
});
var followerSets = results.following.map(function (uid) { await Promise.all([
return 'followers:' + uid; db.sortedSetsRemove(followerSets.concat(followingSets), uid),
}); updateCount(following, 'followers:', 'followerCount'),
updateCount(followers, 'following:', 'followingCount'),
async.parallel([ ]);
async.apply(db.sortedSetsRemove, followerSets.concat(followingSets), uid),
async.apply(updateCount, results.following, 'followers:', 'followerCount'),
async.apply(updateCount, results.followers, 'following:', 'followingCount'),
], callback);
});
} }
}; };

Loading…
Cancel
Save