You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1054 lines
26 KiB
JavaScript
1054 lines
26 KiB
JavaScript
11 years ago
|
var bcrypt = require('bcrypt'),
|
||
12 years ago
|
async = require('async'),
|
||
11 years ago
|
emailjs = require('emailjs'),
|
||
12 years ago
|
nconf = require('nconf'),
|
||
11 years ago
|
winston = require('winston'),
|
||
11 years ago
|
gravatar = require('gravatar'),
|
||
11 years ago
|
check = require('validator').check,
|
||
|
sanitize = require('validator').sanitize,
|
||
11 years ago
|
|
||
11 years ago
|
utils = require('./../public/src/utils'),
|
||
11 years ago
|
plugins = require('./plugins'),
|
||
11 years ago
|
db = require('./database'),
|
||
11 years ago
|
meta = require('./meta'),
|
||
|
emailjsServer = emailjs.server.connect(meta.config['email:smtp:host'] || '127.0.0.1'),
|
||
11 years ago
|
groups = require('./groups'),
|
||
11 years ago
|
notifications = require('./notifications'),
|
||
|
topics = require('./topics');
|
||
12 years ago
|
|
||
11 years ago
|
|
||
12 years ago
|
(function(User) {
|
||
11 years ago
|
'use strict';
|
||
12 years ago
|
User.create = function(username, password, email, callback) {
|
||
|
var userslug = utils.slugify(username);
|
||
|
|
||
12 years ago
|
username = username.trim();
|
||
11 years ago
|
if (email !== undefined) {
|
||
|
email = email.trim();
|
||
|
}
|
||
12 years ago
|
|
||
|
async.parallel([
|
||
|
function(next) {
|
||
11 years ago
|
if (email !== undefined) {
|
||
|
next(!utils.isEmailValid(email) ? new Error('Invalid Email!') : null);
|
||
11 years ago
|
} else {
|
||
11 years ago
|
next();
|
||
|
}
|
||
12 years ago
|
},
|
||
|
function(next) {
|
||
11 years ago
|
next((!utils.isUserNameValid(username) || !userslug) ? new Error('Invalid Username!') : null);
|
||
12 years ago
|
},
|
||
|
function(next) {
|
||
11 years ago
|
if (password !== undefined) {
|
||
|
next(!utils.isPasswordValid(password) ? new Error('Invalid Password!') : null);
|
||
11 years ago
|
} else {
|
||
11 years ago
|
next();
|
||
|
}
|
||
12 years ago
|
},
|
||
|
function(next) {
|
||
|
User.exists(userslug, function(exists) {
|
||
|
next(exists ? new Error('Username taken!') : null);
|
||
|
});
|
||
|
},
|
||
|
function(next) {
|
||
12 years ago
|
if (email !== undefined) {
|
||
12 years ago
|
User.isEmailAvailable(email, function(err, available) {
|
||
11 years ago
|
if (err) {
|
||
12 years ago
|
return next(err);
|
||
11 years ago
|
}
|
||
12 years ago
|
next(!available ? new Error('Email taken!') : null);
|
||
|
});
|
||
11 years ago
|
} else {
|
||
|
next();
|
||
|
}
|
||
12 years ago
|
}
|
||
12 years ago
|
], function(err, results) {
|
||
11 years ago
|
if (err) {
|
||
11 years ago
|
return callback(err);
|
||
11 years ago
|
}
|
||
12 years ago
|
|
||
11 years ago
|
db.incrObjectField('global', 'nextUid', function(err, uid) {
|
||
11 years ago
|
if(err) {
|
||
|
return callback(err);
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
var gravatar = User.createGravatarURLFromEmail(email);
|
||
12 years ago
|
var timestamp = Date.now();
|
||
12 years ago
|
|
||
11 years ago
|
db.setObject('user:' + uid, {
|
||
12 years ago
|
'uid': uid,
|
||
12 years ago
|
'username': username,
|
||
|
'userslug': userslug,
|
||
12 years ago
|
'fullname': '',
|
||
12 years ago
|
'location': '',
|
||
|
'birthday': '',
|
||
|
'website': '',
|
||
|
'email': email || '',
|
||
|
'signature': '',
|
||
|
'joindate': timestamp,
|
||
12 years ago
|
'picture': gravatar,
|
||
12 years ago
|
'gravatarpicture': gravatar,
|
||
12 years ago
|
'uploadedpicture': '',
|
||
12 years ago
|
'profileviews': 0,
|
||
12 years ago
|
'reputation': 0,
|
||
|
'postcount': 0,
|
||
|
'lastposttime': 0,
|
||
12 years ago
|
'banned': 0,
|
||
12 years ago
|
'showemail': 0
|
||
12 years ago
|
});
|
||
12 years ago
|
|
||
11 years ago
|
db.setObjectField('username:uid', username, uid);
|
||
|
db.setObjectField('userslug:uid', userslug, uid);
|
||
12 years ago
|
|
||
12 years ago
|
if (email !== undefined) {
|
||
11 years ago
|
db.setObjectField('email:uid', email, uid);
|
||
11 years ago
|
User.sendConfirmationEmail(email);
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
11 years ago
|
plugins.fireHook('action:user.create', {uid: uid, username: username, email: email, picture: gravatar, timestamp: timestamp});
|
||
11 years ago
|
db.incrObjectField('global', 'userCount');
|
||
12 years ago
|
|
||
11 years ago
|
db.sortedSetAdd('users:joindate', timestamp, uid);
|
||
|
db.sortedSetAdd('users:postcount', 0, uid);
|
||
|
db.sortedSetAdd('users:reputation', 0, uid);
|
||
12 years ago
|
|
||
11 years ago
|
db.searchIndex('user', username, uid);
|
||
12 years ago
|
|
||
12 years ago
|
if (password !== undefined) {
|
||
12 years ago
|
User.hashPassword(password, function(err, hash) {
|
||
12 years ago
|
User.setUserField(uid, 'password', hash);
|
||
12 years ago
|
callback(null, uid);
|
||
12 years ago
|
});
|
||
11 years ago
|
} else {
|
||
|
callback(null, uid);
|
||
|
}
|
||
12 years ago
|
});
|
||
|
});
|
||
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.ban = function(uid, callback) {
|
||
|
User.setUserField(uid, 'banned', 1, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.unban = function(uid, callback) {
|
||
12 years ago
|
User.setUserField(uid, 'banned', 0, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.getUserField = function(uid, field, callback) {
|
||
11 years ago
|
db.getObjectField('user:' + uid, field, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.getUserFields = function(uid, fields, callback) {
|
||
11 years ago
|
db.getObjectFields('user:' + uid, fields, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.getMultipleUserFields = function(uids, fields, callback) {
|
||
12 years ago
|
if (uids.length === 0) {
|
||
12 years ago
|
return callback(null, []);
|
||
12 years ago
|
}
|
||
|
|
||
12 years ago
|
var returnData = [];
|
||
|
|
||
11 years ago
|
var uuids = uids.filter(function(value, index, self) {
|
||
12 years ago
|
return self.indexOf(value) === index;
|
||
|
});
|
||
12 years ago
|
|
||
12 years ago
|
function iterator(uid, next) {
|
||
12 years ago
|
User.getUserFields(uid, fields, function(err, userData) {
|
||
11 years ago
|
if (err) {
|
||
12 years ago
|
return next(err);
|
||
11 years ago
|
}
|
||
12 years ago
|
returnData.push(userData);
|
||
12 years ago
|
next(null);
|
||
12 years ago
|
});
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
async.eachSeries(uuids, iterator, function(err) {
|
||
12 years ago
|
callback(err, returnData);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.getUserData = function(uid, callback) {
|
||
11 years ago
|
db.getObject('user:' + uid, function(err, data) {
|
||
|
if(err) {
|
||
|
return callback(err);
|
||
|
}
|
||
12 years ago
|
|
||
11 years ago
|
if (data && data.password) {
|
||
|
delete data.password;
|
||
12 years ago
|
}
|
||
12 years ago
|
callback(err, data);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.updateProfile = function(uid, data, callback) {
|
||
12 years ago
|
|
||
12 years ago
|
var fields = ['email', 'fullname', 'website', 'location', 'birthday', 'signature'];
|
||
12 years ago
|
var returnData = {
|
||
|
success: false
|
||
|
};
|
||
12 years ago
|
|
||
12 years ago
|
function isSignatureValid(next) {
|
||
11 years ago
|
if (data.signature !== undefined && data.signature.length > 150) {
|
||
12 years ago
|
next({
|
||
|
error: 'Signature can\'t be longer than 150 characters!'
|
||
|
}, false);
|
||
12 years ago
|
} else {
|
||
12 years ago
|
next(null, true);
|
||
|
}
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
function isEmailAvailable(next) {
|
||
11 years ago
|
if (!data.email) {
|
||
12 years ago
|
return next(null, true);
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
User.getUserField(uid, 'email', function(err, email) {
|
||
11 years ago
|
if (email !== data.email) {
|
||
|
User.isEmailAvailable(data.email, function(err, available) {
|
||
|
if (err) {
|
||
12 years ago
|
return next(err, null);
|
||
11 years ago
|
}
|
||
12 years ago
|
if (!available) {
|
||
|
next({
|
||
|
error: 'Email not available!'
|
||
|
}, false);
|
||
12 years ago
|
} else {
|
||
12 years ago
|
next(null, true);
|
||
12 years ago
|
}
|
||
|
});
|
||
|
} else {
|
||
12 years ago
|
next(null, true);
|
||
12 years ago
|
}
|
||
|
});
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
async.series([isSignatureValid, isEmailAvailable], function(err, results) {
|
||
12 years ago
|
if (err) {
|
||
12 years ago
|
callback(err, returnData);
|
||
12 years ago
|
} else {
|
||
12 years ago
|
async.each(fields, updateField, function(err) {
|
||
12 years ago
|
if (err) {
|
||
12 years ago
|
callback(err, returnData);
|
||
12 years ago
|
} else {
|
||
|
returnData.success = true;
|
||
12 years ago
|
callback(null, returnData);
|
||
12 years ago
|
}
|
||
|
});
|
||
12 years ago
|
}
|
||
|
});
|
||
12 years ago
|
|
||
12 years ago
|
function updateField(field, next) {
|
||
11 years ago
|
if (data[field] !== undefined && typeof data[field] === 'string') {
|
||
11 years ago
|
data[field] = data[field].trim();
|
||
|
data[field] = sanitize(data[field]).escape();
|
||
|
|
||
12 years ago
|
if (field === 'email') {
|
||
12 years ago
|
var gravatarpicture = User.createGravatarURLFromEmail(data[field]);
|
||
|
User.setUserField(uid, 'gravatarpicture', gravatarpicture);
|
||
12 years ago
|
User.getUserFields(uid, ['email', 'picture', 'uploadedpicture'], function(err, userData) {
|
||
11 years ago
|
if (err) {
|
||
12 years ago
|
return next(err);
|
||
11 years ago
|
}
|
||
12 years ago
|
|
||
11 years ago
|
db.deleteObjectField('email:uid', userData.email);
|
||
|
db.setObjectField('email:uid', data.email, uid);
|
||
12 years ago
|
User.setUserField(uid, field, data[field]);
|
||
12 years ago
|
if (userData.picture !== userData.uploadedpicture) {
|
||
12 years ago
|
returnData.picture = gravatarpicture;
|
||
|
User.setUserField(uid, 'picture', gravatarpicture);
|
||
|
}
|
||
|
returnData.gravatarpicture = gravatarpicture;
|
||
12 years ago
|
next(null);
|
||
12 years ago
|
});
|
||
|
return;
|
||
12 years ago
|
} else if (field === 'signature') {
|
||
12 years ago
|
data[field] = utils.strip_tags(data[field]);
|
||
11 years ago
|
} else if (field === 'website') {
|
||
|
if(data[field].substr(0, 7) !== 'http://' && data[field].substr(0, 8) !== 'https://') {
|
||
|
data[field] = 'http://' + data[field];
|
||
|
}
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
User.setUserField(uid, field, data[field]);
|
||
12 years ago
|
|
||
12 years ago
|
next(null);
|
||
12 years ago
|
} else {
|
||
12 years ago
|
next(null);
|
||
12 years ago
|
}
|
||
12 years ago
|
}
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.isEmailAvailable = function(email, callback) {
|
||
11 years ago
|
db.isObjectField('email:uid', email, function(err, exists) {
|
||
12 years ago
|
callback(err, !exists);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.changePassword = function(uid, data, callback) {
|
||
12 years ago
|
if (!utils.isPasswordValid(data.newPassword)) {
|
||
|
return callback({
|
||
|
error: 'Invalid password!'
|
||
|
});
|
||
12 years ago
|
}
|
||
|
|
||
12 years ago
|
User.getUserField(uid, 'password', function(err, user_password) {
|
||
12 years ago
|
bcrypt.compare(data.currentPassword, user_password, function(err, res) {
|
||
12 years ago
|
if (err) {
|
||
12 years ago
|
return callback(err);
|
||
12 years ago
|
}
|
||
|
|
||
|
if (res) {
|
||
12 years ago
|
User.hashPassword(data.newPassword, function(err, hash) {
|
||
12 years ago
|
User.setUserField(uid, 'password', hash);
|
||
|
|
||
12 years ago
|
callback(null);
|
||
12 years ago
|
});
|
||
|
} else {
|
||
12 years ago
|
callback({
|
||
|
error: 'Your current password is not correct!'
|
||
|
});
|
||
12 years ago
|
}
|
||
|
});
|
||
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.setUserField = function(uid, field, value, callback) {
|
||
11 years ago
|
db.setObjectField('user:' + uid, field, value, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.setUserFields = function(uid, data, callback) {
|
||
11 years ago
|
db.setObject('user:' + uid, data, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.incrementUserFieldBy = function(uid, field, value, callback) {
|
||
11 years ago
|
db.incrObjectFieldBy('user:' + uid, field, value, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.decrementUserFieldBy = function(uid, field, value, callback) {
|
||
11 years ago
|
db.incrObjectFieldBy('user:' + uid, field, -value, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.getUsers = function(set, start, stop, callback) {
|
||
12 years ago
|
var data = [];
|
||
12 years ago
|
|
||
11 years ago
|
db.getSortedSetRevRange(set, start, stop, function(err, uids) {
|
||
12 years ago
|
if (err) {
|
||
12 years ago
|
return callback(err, null);
|
||
|
}
|
||
12 years ago
|
|
||
11 years ago
|
function iterator(uid, callback) {
|
||
|
User.getUserData(uid, function(err, userData) {
|
||
11 years ago
|
User.isAdministrator(uid, function(err, isAdmin) {
|
||
11 years ago
|
if (userData) {
|
||
|
userData.administrator = isAdmin?"1":"0";
|
||
|
data.push(userData);
|
||
|
}
|
||
|
callback(null);
|
||
11 years ago
|
});
|
||
|
});
|
||
11 years ago
|
}
|
||
|
|
||
|
async.eachSeries(uids, iterator, function(err) {
|
||
|
callback(err, data);
|
||
12 years ago
|
});
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.createGravatarURLFromEmail = function(email) {
|
||
12 years ago
|
var options = {
|
||
|
size: '128',
|
||
|
default: 'identicon',
|
||
|
rating: 'pg'
|
||
11 years ago
|
},
|
||
|
https = nconf.get('https');
|
||
12 years ago
|
|
||
12 years ago
|
if (!email) {
|
||
12 years ago
|
email = '';
|
||
|
options.forcedefault = 'y';
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
11 years ago
|
return gravatar.url(email, options, https);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.hashPassword = function(password, callback) {
|
||
12 years ago
|
if (!password) {
|
||
12 years ago
|
callback(password);
|
||
|
return;
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
bcrypt.genSalt(nconf.get('bcrypt_rounds'), function(err, salt) {
|
||
12 years ago
|
bcrypt.hash(password, salt, callback);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.reIndexAll = function(callback) {
|
||
|
User.getUsers('users:joindate', 0, -1, function(err, usersData) {
|
||
12 years ago
|
if (err) {
|
||
12 years ago
|
return callback(err, null);
|
||
|
}
|
||
|
|
||
|
function reIndexUser(uid, username) {
|
||
11 years ago
|
db.searchRemove('user', uid, function() {
|
||
|
db.searchIndex('user', username, uid);
|
||
11 years ago
|
});
|
||
12 years ago
|
}
|
||
|
|
||
12 years ago
|
for (var i = 0; i < usersData.length; ++i) {
|
||
12 years ago
|
reIndexUser(usersData[i].uid, usersData[i].username);
|
||
|
}
|
||
|
callback(null, 1);
|
||
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.search = function(username, callback) {
|
||
12 years ago
|
if (!username) {
|
||
12 years ago
|
callback([]);
|
||
|
return;
|
||
|
}
|
||
11 years ago
|
db.search('user', username, function(err, uids) {
|
||
12 years ago
|
if (err) {
|
||
12 years ago
|
console.log(err);
|
||
12 years ago
|
return;
|
||
|
}
|
||
12 years ago
|
if (uids && uids.length) {
|
||
12 years ago
|
User.getDataForUsers(uids, function(userdata) {
|
||
|
callback(userdata);
|
||
|
});
|
||
|
} else {
|
||
|
callback([]);
|
||
12 years ago
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.onNewPostMade = function(uid, tid, pid, timestamp) {
|
||
12 years ago
|
User.addPostIdToUser(uid, pid);
|
||
12 years ago
|
|
||
12 years ago
|
User.incrementUserFieldBy(uid, 'postcount', 1, function(err, newpostcount) {
|
||
11 years ago
|
db.sortedSetAdd('users:postcount', newpostcount, uid);
|
||
12 years ago
|
});
|
||
12 years ago
|
|
||
12 years ago
|
User.setUserField(uid, 'lastposttime', timestamp);
|
||
|
|
||
|
User.sendPostNotificationToFollowers(uid, tid, pid);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.addPostIdToUser = function(uid, pid) {
|
||
11 years ago
|
db.listPrepend('uid:' + uid + ':posts', pid);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.addTopicIdToUser = function(uid, tid) {
|
||
11 years ago
|
db.listPrepend('uid:' + uid + ':topics', tid);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getPostIds = function(uid, start, stop, callback) {
|
||
|
db.getListRange('uid:' + uid + ':posts', start, stop, function(err, pids) {
|
||
12 years ago
|
if (!err) {
|
||
11 years ago
|
if (pids && pids.length) {
|
||
12 years ago
|
callback(pids);
|
||
11 years ago
|
} else {
|
||
12 years ago
|
callback([]);
|
||
11 years ago
|
}
|
||
12 years ago
|
} else {
|
||
12 years ago
|
console.log(err);
|
||
|
callback([]);
|
||
|
}
|
||
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
|
||
12 years ago
|
|
||
12 years ago
|
User.follow = function(uid, followid, callback) {
|
||
11 years ago
|
db.setAdd('following:' + uid, followid, function(err, data) {
|
||
12 years ago
|
if (!err) {
|
||
11 years ago
|
db.setAdd('followers:' + followid, uid, function(err, data) {
|
||
12 years ago
|
if (!err) {
|
||
12 years ago
|
callback(true);
|
||
|
} else {
|
||
|
console.log(err);
|
||
|
callback(false);
|
||
|
}
|
||
12 years ago
|
});
|
||
12 years ago
|
} else {
|
||
12 years ago
|
console.log(err);
|
||
12 years ago
|
callback(false);
|
||
12 years ago
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.unfollow = function(uid, unfollowid, callback) {
|
||
11 years ago
|
db.setRemove('following:' + uid, unfollowid, function(err, data) {
|
||
12 years ago
|
if (!err) {
|
||
11 years ago
|
db.setRemove('followers:' + unfollowid, uid, function(err, data) {
|
||
12 years ago
|
callback(data);
|
||
|
});
|
||
12 years ago
|
} else {
|
||
12 years ago
|
console.log(err);
|
||
12 years ago
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.getFollowing = function(uid, callback) {
|
||
11 years ago
|
db.getSetMembers('following:' + uid, function(err, userIds) {
|
||
12 years ago
|
if (!err) {
|
||
12 years ago
|
User.getDataForUsers(userIds, callback);
|
||
12 years ago
|
} else {
|
||
|
console.log(err);
|
||
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.getFollowers = function(uid, callback) {
|
||
11 years ago
|
db.getSetMembers('followers:' + uid, function(err, userIds) {
|
||
12 years ago
|
if (!err) {
|
||
12 years ago
|
User.getDataForUsers(userIds, callback);
|
||
12 years ago
|
} else {
|
||
12 years ago
|
console.log(err);
|
||
12 years ago
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.getFollowingCount = function(uid, callback) {
|
||
11 years ago
|
db.getSetMembers('following:' + uid, function(err, userIds) {
|
||
11 years ago
|
if (err) {
|
||
12 years ago
|
console.log(err);
|
||
11 years ago
|
} else {
|
||
|
userIds = userIds.filter(function(value) {
|
||
11 years ago
|
return parseInt(value, 10) !== 0;
|
||
11 years ago
|
});
|
||
|
callback(userIds.length);
|
||
12 years ago
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.getFollowerCount = function(uid, callback) {
|
||
11 years ago
|
db.getSetMembers('followers:' + uid, function(err, userIds) {
|
||
11 years ago
|
if(err) {
|
||
12 years ago
|
console.log(err);
|
||
11 years ago
|
} else {
|
||
|
userIds = userIds.filter(function(value) {
|
||
11 years ago
|
return parseInt(value, 10) !== 0;
|
||
11 years ago
|
});
|
||
|
callback(userIds.length);
|
||
12 years ago
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.getDataForUsers = function(uids, callback) {
|
||
12 years ago
|
var returnData = [];
|
||
|
|
||
12 years ago
|
if (!uids || !Array.isArray(uids) || uids.length === 0) {
|
||
12 years ago
|
callback(returnData);
|
||
|
return;
|
||
|
}
|
||
|
|
||
12 years ago
|
function iterator(uid, callback) {
|
||
11 years ago
|
if(parseInt(uid, 10) === 0) {
|
||
11 years ago
|
return callback(null);
|
||
11 years ago
|
}
|
||
11 years ago
|
|
||
12 years ago
|
User.getUserData(uid, function(err, userData) {
|
||
12 years ago
|
returnData.push(userData);
|
||
12 years ago
|
|
||
12 years ago
|
callback(null);
|
||
12 years ago
|
});
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
async.eachSeries(uids, iterator, function(err) {
|
||
12 years ago
|
callback(returnData);
|
||
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.sendPostNotificationToFollowers = function(uid, tid, pid) {
|
||
12 years ago
|
User.getUserField(uid, 'username', function(err, username) {
|
||
11 years ago
|
db.getSetMembers('followers:' + uid, function(err, followers) {
|
||
12 years ago
|
topics.getTopicField(tid, 'slug', function(err, slug) {
|
||
12 years ago
|
var message = '<strong>' + username + '</strong> made a new post';
|
||
12 years ago
|
|
||
11 years ago
|
notifications.create(message, nconf.get('relative_path') + '/topic/' + slug + '#' + pid, 'topic:' + tid, function(nid) {
|
||
12 years ago
|
notifications.push(nid, followers);
|
||
12 years ago
|
});
|
||
12 years ago
|
});
|
||
|
});
|
||
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.isFollowing = function(uid, theirid, callback) {
|
||
11 years ago
|
db.isSetMember('following:' + uid, theirid, function(err, isMember) {
|
||
12 years ago
|
if (!err) {
|
||
11 years ago
|
callback(isMember);
|
||
12 years ago
|
} else {
|
||
12 years ago
|
console.log(err);
|
||
12 years ago
|
}
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.exists = function(userslug, callback) {
|
||
11 years ago
|
User.getUidByUserslug(userslug, function(err, exists) {
|
||
12 years ago
|
callback( !! exists);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.count = function(socket) {
|
||
11 years ago
|
db.getObjectField('global', 'userCount', function(err, count) {
|
||
11 years ago
|
if(err) {
|
||
|
return;
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
socket.emit('user.count', {
|
||
|
count: count ? count : 0
|
||
|
});
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUidByUsername = function(username, callback) {
|
||
11 years ago
|
db.getObjectField('username:uid', username, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUidByUserslug = function(userslug, callback) {
|
||
11 years ago
|
db.getObjectField('userslug:uid', userslug, callback);
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUsernamesByUids = function(uids, callback) {
|
||
12 years ago
|
var usernames = [];
|
||
12 years ago
|
|
||
12 years ago
|
if (!Array.isArray(uids)) {
|
||
|
return callback([]);
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
function iterator(uid, callback) {
|
||
12 years ago
|
User.getUserField(uid, 'username', function(err, username) {
|
||
12 years ago
|
usernames.push(username);
|
||
12 years ago
|
callback(null);
|
||
12 years ago
|
});
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
async.eachSeries(uids, iterator, function(err) {
|
||
12 years ago
|
callback(usernames);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUserSlugsByUids = function(uids, callback) {
|
||
12 years ago
|
var userslugs = [];
|
||
|
|
||
12 years ago
|
if (!Array.isArray(uids)) {
|
||
|
return callback([]);
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
function iterator(uid, callback) {
|
||
12 years ago
|
User.getUserField(uid, 'userslug', function(err, userslug) {
|
||
12 years ago
|
userslugs.push(userslug);
|
||
|
callback(null);
|
||
|
});
|
||
|
}
|
||
|
|
||
12 years ago
|
async.eachSeries(uids, iterator, function(err) {
|
||
12 years ago
|
callback(userslugs);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUidByEmail = function(email, callback) {
|
||
11 years ago
|
db.getObjectField('email:uid', email, function(err, data) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return callback(err);
|
||
12 years ago
|
}
|
||
11 years ago
|
callback(null, data);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUidByTwitterId = function(twid, callback) {
|
||
11 years ago
|
db.getObjectField('twid:uid', twid, function(err, uid) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return callback(err);
|
||
12 years ago
|
}
|
||
11 years ago
|
callback(null, uid);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUidByGoogleId = function(gplusid, callback) {
|
||
11 years ago
|
db.getObjectField('gplusid:uid', gplusid, function(err, uid) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return callback(err);
|
||
12 years ago
|
}
|
||
11 years ago
|
callback(null, uid);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.getUidByFbid = function(fbid, callback) {
|
||
11 years ago
|
db.getObjectField('fbid:uid', fbid, function(err, uid) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return callback(err);
|
||
12 years ago
|
}
|
||
11 years ago
|
callback(null, uid);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.isModerator = function(uid, cid, callback) {
|
||
11 years ago
|
db.isSetMember('cid:' + cid + ':moderators', uid, function(err, exists) {
|
||
|
if(err) {
|
||
|
return calback(err);
|
||
|
}
|
||
11 years ago
|
callback(err, exists);
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
|
User.isAdministrator = function(uid, callback) {
|
||
11 years ago
|
groups.getGidFromName('Administrators', function(err, gid) {
|
||
|
groups.isMember(uid, gid, function(err, isAdmin) {
|
||
11 years ago
|
callback(err, isAdmin);
|
||
12 years ago
|
});
|
||
12 years ago
|
});
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
12 years ago
|
User.reset = {
|
||
|
validate: function(socket, code, callback) {
|
||
12 years ago
|
|
||
12 years ago
|
if (typeof callback !== 'function') {
|
||
|
callback = null;
|
||
|
}
|
||
12 years ago
|
|
||
11 years ago
|
db.getObjectField('reset:uid', code, function(err, uid) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return callback(false);
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
|
if (uid !== null) {
|
||
11 years ago
|
db.getObjectField('reset:expiry', code, function(err, expiry) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return callback(false);
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
if (expiry >= +Date.now() / 1000 | 0) {
|
||
12 years ago
|
if (!callback) {
|
||
12 years ago
|
socket.emit('user:reset.valid', {
|
||
|
valid: true
|
||
|
});
|
||
12 years ago
|
} else {
|
||
|
callback(true);
|
||
|
}
|
||
12 years ago
|
} else {
|
||
|
// Expired, delete from db
|
||
11 years ago
|
db.deleteObjectField('reset:uid', code);
|
||
|
db.deleteObjectField('reset:expiry', code);
|
||
12 years ago
|
if (!callback) {
|
||
12 years ago
|
socket.emit('user:reset.valid', {
|
||
|
valid: false
|
||
|
});
|
||
12 years ago
|
} else {
|
||
|
callback(false);
|
||
|
}
|
||
12 years ago
|
}
|
||
|
});
|
||
|
} else {
|
||
12 years ago
|
if (!callback) {
|
||
12 years ago
|
socket.emit('user:reset.valid', {
|
||
|
valid: false
|
||
|
});
|
||
12 years ago
|
} else {
|
||
|
callback(false);
|
||
|
}
|
||
12 years ago
|
}
|
||
|
});
|
||
|
},
|
||
|
send: function(socket, email) {
|
||
11 years ago
|
User.getUidByEmail(email, function(err, uid) {
|
||
12 years ago
|
if (uid !== null) {
|
||
|
// Generate a new reset code
|
||
|
var reset_code = utils.generateUUID();
|
||
11 years ago
|
db.setObjectField('reset:uid', reset_code, uid);
|
||
|
db.setobjectField('reset:expiry', reset_code, (60 * 60) + new Date() / 1000 | 0); // Active for one hour
|
||
12 years ago
|
|
||
12 years ago
|
var reset_link = nconf.get('url') + 'reset/' + reset_code,
|
||
12 years ago
|
reset_email = global.templates['emails/reset'].parse({
|
||
|
'RESET_LINK': reset_link
|
||
|
}),
|
||
|
reset_email_plaintext = global.templates['emails/reset_plaintext'].parse({
|
||
|
'RESET_LINK': reset_link
|
||
|
});
|
||
12 years ago
|
|
||
|
var message = emailjs.message.create({
|
||
|
text: reset_email_plaintext,
|
||
11 years ago
|
from: meta.config['email:from'] ? meta.config['email:from'] : '[email protected]',
|
||
12 years ago
|
to: email,
|
||
|
subject: 'Password Reset Requested',
|
||
12 years ago
|
attachment: [{
|
||
|
data: reset_email,
|
||
|
alternative: true
|
||
|
}]
|
||
12 years ago
|
});
|
||
12 years ago
|
|
||
12 years ago
|
emailjsServer.send(message, function(err, success) {
|
||
|
if (err === null) {
|
||
|
socket.emit('user.send_reset', {
|
||
|
status: "ok",
|
||
|
message: "code-sent",
|
||
|
email: email
|
||
|
});
|
||
|
} else {
|
||
|
socket.emit('user.send_reset', {
|
||
|
status: "error",
|
||
|
message: "send-failed"
|
||
|
});
|
||
11 years ago
|
winston.err(err);
|
||
12 years ago
|
}
|
||
|
});
|
||
|
} else {
|
||
|
socket.emit('user.send_reset', {
|
||
|
status: "error",
|
||
|
message: "invalid-email",
|
||
|
email: email
|
||
|
});
|
||
|
}
|
||
|
});
|
||
|
},
|
||
|
commit: function(socket, code, password) {
|
||
12 years ago
|
this.validate(socket, code, function(validated) {
|
||
12 years ago
|
if (validated) {
|
||
11 years ago
|
db.getObjectField('reset:uid', code, function(err, uid) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return;
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
12 years ago
|
User.hashPassword(password, function(err, hash) {
|
||
12 years ago
|
User.setUserField(uid, 'password', hash);
|
||
|
});
|
||
|
|
||
11 years ago
|
db.deleteObjectField('reset:uid', code);
|
||
|
db.deleteObjectField('reset:expiry', code);
|
||
12 years ago
|
|
||
12 years ago
|
socket.emit('user:reset.commit', {
|
||
|
status: 'ok'
|
||
|
});
|
||
12 years ago
|
});
|
||
|
}
|
||
|
});
|
||
|
}
|
||
11 years ago
|
};
|
||
12 years ago
|
|
||
11 years ago
|
User.sendConfirmationEmail = function(email) {
|
||
|
if (meta.config['email:smtp:host'] && meta.config['email:smtp:port'] && meta.config['email:from']) {
|
||
|
var confirm_code = utils.generateUUID(),
|
||
|
confirm_link = nconf.get('url') + 'confirm/' + confirm_code,
|
||
|
confirm_email = global.templates['emails/header'] + global.templates['emails/email_confirm'].parse({
|
||
|
'CONFIRM_LINK': confirm_link
|
||
|
}) + global.templates['emails/footer'],
|
||
|
confirm_email_plaintext = global.templates['emails/email_confirm_plaintext'].parse({
|
||
|
'CONFIRM_LINK': confirm_link
|
||
|
});
|
||
|
|
||
|
// Email confirmation code
|
||
|
var expiry_time = Date.now() / 1000 + 60 * 60 * 2;
|
||
|
|
||
|
db.setObjectField('email:confirm', email, confirm_code);
|
||
|
|
||
|
db.setObjectField('confirm:email', confirm_code, email);
|
||
|
db.setObjectField('confirm:email', confirm_code + ':expire', expiry_time);
|
||
|
|
||
|
// Send intro email w/ confirm code
|
||
|
var message = emailjs.message.create({
|
||
|
text: confirm_email_plaintext,
|
||
|
from: meta.config['email:from'] || '[email protected]',
|
||
|
to: email,
|
||
|
subject: '[NodeBB] Registration Email Verification',
|
||
|
attachment: [{
|
||
|
data: confirm_email,
|
||
|
alternative: true
|
||
|
}]
|
||
|
});
|
||
|
|
||
|
emailjsServer.send(message, function(err, success) {
|
||
|
if (err) {
|
||
|
console.log(err);
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
|
||
12 years ago
|
User.email = {
|
||
|
exists: function(socket, email, callback) {
|
||
11 years ago
|
User.getUidByEmail(email, function(err, exists) {
|
||
12 years ago
|
exists = !! exists;
|
||
12 years ago
|
if (typeof callback !== 'function') {
|
||
12 years ago
|
socket.emit('user.email.exists', {
|
||
|
exists: exists
|
||
|
});
|
||
12 years ago
|
} else {
|
||
|
callback(exists);
|
||
|
}
|
||
12 years ago
|
});
|
||
|
},
|
||
|
confirm: function(code, callback) {
|
||
11 years ago
|
db.getObjectFields('confirm:email', [code, code + ':expire'], function(err, data) {
|
||
12 years ago
|
if (err) {
|
||
11 years ago
|
return callback({
|
||
|
status:'error'
|
||
|
});
|
||
|
}
|
||
|
|
||
|
var email = data.email;
|
||
|
var expiry = data[code + ':expire'];
|
||
|
if (parseInt(expiry, 10) >= Date.now() / 1000) {
|
||
|
|
||
|
db.deleteObjectField('confirm:email', code);
|
||
|
db.deleteObjectField('confirm:email', code + ':expire');
|
||
|
|
||
|
return callback({
|
||
|
status: 'expired'
|
||
|
});
|
||
12 years ago
|
}
|
||
12 years ago
|
|
||
|
if (email !== null) {
|
||
11 years ago
|
db.setObjectField('email:confirm', email, true);
|
||
|
|
||
|
db.deleteObjectField('confirm:email', code);
|
||
|
db.deleteObjectField('confirm:email', code + ':expire');
|
||
12 years ago
|
callback({
|
||
|
status: 'ok'
|
||
|
});
|
||
12 years ago
|
} else {
|
||
12 years ago
|
callback({
|
||
|
status: 'not_ok'
|
||
|
});
|
||
12 years ago
|
}
|
||
|
});
|
||
|
}
|
||
12 years ago
|
};
|
||
|
|
||
12 years ago
|
User.notifications = {
|
||
|
get: function(uid, callback) {
|
||
12 years ago
|
var maxNotifs = 15;
|
||
12 years ago
|
|
||
12 years ago
|
async.parallel({
|
||
|
unread: function(next) {
|
||
11 years ago
|
db.getSortedSetRevRange('uid:' + uid + ':notifications:unread', 0, 10, function(err, nids) {
|
||
12 years ago
|
// @todo handle err
|
||
12 years ago
|
var unread = [];
|
||
12 years ago
|
|
||
|
// Cap the number of notifications returned
|
||
11 years ago
|
if (nids.length > maxNotifs) {
|
||
|
nids.length = maxNotifs;
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
if (nids && nids.length > 0) {
|
||
|
async.eachSeries(nids, function(nid, next) {
|
||
11 years ago
|
notifications.get(nid, uid, function(notif_data) {
|
||
11 years ago
|
// If the notification could not be found, silently drop it
|
||
|
if (notif_data) {
|
||
|
unread.push(notif_data);
|
||
|
} else {
|
||
11 years ago
|
db.sortedSetRemove('uid:' + uid + ':notifications:unread', nid);
|
||
11 years ago
|
}
|
||
|
|
||
12 years ago
|
next();
|
||
|
});
|
||
|
}, function(err) {
|
||
|
next(null, unread);
|
||
|
});
|
||
12 years ago
|
} else {
|
||
|
next(null, unread);
|
||
|
}
|
||
12 years ago
|
});
|
||
|
},
|
||
|
read: function(next) {
|
||
11 years ago
|
db.getSortedSetRevRange('uid:' + uid + ':notifications:read', 0, 10, function(err, nids) {
|
||
12 years ago
|
// @todo handle err
|
||
12 years ago
|
var read = [];
|
||
12 years ago
|
|
||
|
// Cap the number of notifications returned
|
||
11 years ago
|
if (nids.length > maxNotifs) {
|
||
|
nids.length = maxNotifs;
|
||
|
}
|
||
12 years ago
|
|
||
12 years ago
|
if (nids && nids.length > 0) {
|
||
|
async.eachSeries(nids, function(nid, next) {
|
||
11 years ago
|
notifications.get(nid, uid, function(notif_data) {
|
||
11 years ago
|
// If the notification could not be found, silently drop it
|
||
|
if (notif_data) {
|
||
|
read.push(notif_data);
|
||
|
} else {
|
||
11 years ago
|
db.sortedSetRemove('uid:' + uid + ':notifications:read', nid);
|
||
11 years ago
|
}
|
||
|
|
||
12 years ago
|
next();
|
||
|
});
|
||
|
}, function(err) {
|
||
|
next(null, read);
|
||
|
});
|
||
12 years ago
|
} else {
|
||
|
next(null, read);
|
||
|
}
|
||
12 years ago
|
});
|
||
|
}
|
||
|
}, function(err, notifications) {
|
||
12 years ago
|
// Limit the number of notifications to `maxNotifs`, prioritising unread notifications
|
||
|
if (notifications.read.length + notifications.unread.length > maxNotifs) {
|
||
|
notifications.read.length = maxNotifs - notifications.unread.length;
|
||
|
}
|
||
|
|
||
12 years ago
|
callback(notifications);
|
||
|
});
|
||
12 years ago
|
},
|
||
11 years ago
|
getAll: function(uid, limit, before, callback) {
|
||
|
var now = new Date();
|
||
|
|
||
11 years ago
|
if (!limit || parseInt(limit,10) <= 0) {
|
||
|
limit = 25;
|
||
|
}
|
||
|
if (before) {
|
||
|
before = new Date(parseInt(before, 10));
|
||
|
}
|
||
11 years ago
|
|
||
11 years ago
|
var args1 = ['uid:' + uid + ':notifications:read', before ? before.getTime(): now.getTime(), -Infinity, 'LIMIT', 0, limit];
|
||
|
var args2 = ['uid:' + uid + ':notifications:unread', before ? before.getTime(): now.getTime(), -Infinity, 'LIMIT', 0, limit];
|
||
11 years ago
|
|
||
11 years ago
|
db.getSortedSetRevRangeByScore(args1, function(err, results1) {
|
||
|
db.getSortedSetRevRangeByScore(args2, function(err, results2) {
|
||
|
|
||
|
var nids = results1.concat(results2);
|
||
11 years ago
|
async.map(nids, function(nid, next) {
|
||
11 years ago
|
notifications.get(nid, uid, function(notif_data) {
|
||
11 years ago
|
next(null, notif_data);
|
||
|
});
|
||
|
}, function(err, notifs) {
|
||
11 years ago
|
notifs = notifs.filter(function(notif) {
|
||
11 years ago
|
return notif !== null;
|
||
11 years ago
|
}).sort(function(a, b) {
|
||
11 years ago
|
return parseInt(b.datetime, 10) - parseInt(a.datetime, 10);
|
||
|
}).map(function(notif) {
|
||
|
notif.datetimeISO = new Date(parseInt(notif.datetime, 10)).toISOString();
|
||
11 years ago
|
notif.readClass = !notif.read ? 'unread' : '';
|
||
11 years ago
|
|
||
|
return notif;
|
||
|
});
|
||
|
|
||
|
callback(err, notifs);
|
||
11 years ago
|
});
|
||
11 years ago
|
});
|
||
11 years ago
|
});
|
||
|
|
||
11 years ago
|
},
|
||
12 years ago
|
getUnreadCount: function(uid, callback) {
|
||
11 years ago
|
db.sortedSetCount('uid:' + uid + ':notifications:unread', -Infinity, Infinity, callback);
|
||
12 years ago
|
},
|
||
|
getUnreadByUniqueId: function(uid, uniqueId, callback) {
|
||
11 years ago
|
db.getSortedSetRange('uid:' + uid + ':notifications:unread', 0, -1, function(err, nids) {
|
||
11 years ago
|
|
||
12 years ago
|
async.filter(nids, function(nid, next) {
|
||
11 years ago
|
notifications.get(nid, uid, function(notifObj) {
|
||
11 years ago
|
if(!notifObj) {
|
||
|
next(false);
|
||
|
}
|
||
11 years ago
|
if (notifObj.uniqueId === uniqueId) {
|
||
|
next(true);
|
||
11 years ago
|
} else {
|
||
11 years ago
|
next(false);
|
||
|
}
|
||
12 years ago
|
});
|
||
|
}, function(nids) {
|
||
|
callback(null, nids);
|
||
|
});
|
||
|
});
|
||
12 years ago
|
}
|
||
11 years ago
|
};
|
||
11 years ago
|
}(exports));
|