Files
meanTorrent/modules/users/server/controllers/admin.server.controller.js

859 lines
21 KiB
JavaScript
Raw Permalink Normal View History

'use strict';
/**
* Module dependencies
*/
var path = require('path'),
config = require(path.resolve('./config/config')),
dataLog = require(path.resolve('./config/lib/data-log')),
mongoose = require('mongoose'),
objectId = require('mongodb').ObjectId,
User = mongoose.model('User'),
Peer = mongoose.model('Peer'),
Torrent = mongoose.model('Torrent'),
Complete = mongoose.model('Complete'),
Invitation = mongoose.model('Invitation'),
moment = require('moment'),
2017-10-14 23:15:53 +08:00
async = require('async'),
ScoreLog = mongoose.model('ScoreLog'),
scoreUpdate = require(path.resolve('./config/lib/score')).update,
errorHandler = require(path.resolve('./modules/core/server/controllers/errors.server.controller')),
traceLogCreate = require(path.resolve('./config/lib/tracelog')).create,
history = require(path.resolve('./config/lib/history')),
populateStrings = require(path.resolve('./config/lib/populateStrings'));
const PEERSTATE_SEEDER = 'seeder';
const PEERSTATE_LEECHER = 'leecher';
var traceConfig = config.meanTorrentConfig.trace;
var historyConfig = config.meanTorrentConfig.history;
2017-10-14 23:15:53 +08:00
var mtDebug = require(path.resolve('./config/lib/debug'));
var appConfig = config.meanTorrentConfig.app;
var serverMessage = require(path.resolve('./config/lib/server-message'));
var serverNoticeConfig = config.meanTorrentConfig.serverNotice;
var announceConfig = config.meanTorrentConfig.announce;
var scoreConfig = config.meanTorrentConfig.score;
var inviteConfig = config.meanTorrentConfig.invite;
/**
* Show the current user
*/
exports.read = function (req, res) {
res.json(req.model);
};
/**
* Update a User
*/
exports.update = function (req, res) {
var user = req.model;
var accessChanged = false;
accessChanged = (user.upload_access !== req.body.upload_access);
// For security purposes only merge these parameters
user.firstName = req.body.firstName;
user.lastName = req.body.lastName;
user.displayName = user.firstName + ' ' + user.lastName;
user.roles = req.body.roles;
user.upload_access = req.body.upload_access;
user.remarks = req.body.remarks;
user.save(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
//add server message
if (serverNoticeConfig.action.userUploadAccessChanged.enable && accessChanged) {
serverMessage.addMessage(user._id, serverNoticeConfig.action.userUploadAccessChanged.title, serverNoticeConfig.action.userUploadAccessChanged.content, {
upload_access: user.upload_access === 'review' ? 'UPLOADER.FIELDS_REVIEW' : 'UPLOADER.FIELDS_PASS',
by_name: req.user.displayName,
by_id: req.user._id,
site_name: appConfig.name
});
}
//create trace log
traceLogCreate(req, traceConfig.action.adminUserEdit, {
2017-06-24 13:53:32 +08:00
user: user._id,
firstName: req.body.firstName,
lastName: req.body.lastName,
2018-06-11 15:19:52 +08:00
roles: req.body.roles,
upload_access: req.body.upload_access,
remarks: req.body.remarks
});
});
};
/**
* Delete a user
*/
exports.delete = function (req, res) {
var user = req.model;
user.remove(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
//create trace log
traceLogCreate(req, traceConfig.action.adminUserDelete, {
2017-06-24 13:53:32 +08:00
user: user._id
});
});
};
/**
* List of Users
*/
exports.list = function (req, res) {
2017-10-14 23:15:53 +08:00
var condition = {};
var keysA = [];
var skip = 0;
var limit = 0;
var isVip = false;
var isOper = false;
var isAdmin = false;
var status = undefined;
2017-10-14 23:15:53 +08:00
if (req.query.skip !== undefined) {
skip = parseInt(req.query.skip, 10);
}
if (req.query.limit !== undefined) {
limit = parseInt(req.query.limit, 10);
}
if (req.query.isVip !== undefined) {
isVip = req.query.isVip;
}
if (req.query.isOper !== undefined) {
isOper = req.query.isOper;
}
if (req.query.isAdmin !== undefined) {
isAdmin = req.query.isAdmin;
}
if (req.query.status !== undefined && req.query.status !== 'all') {
status = req.query.status;
}
2017-10-14 23:15:53 +08:00
if (req.query.keys && req.query.keys.length > 0) {
var keysS = req.query.keys + '';
var keysT = keysS.split(' ');
if (keysT.length === 1 && mongoose.Types.ObjectId.isValid(keysT[0])) {
keysA = objectId(keysT[0]);
} else {
keysT.forEach(function (it) {
var ti = new RegExp(it, 'i');
keysA.push(ti);
});
}
2017-10-14 23:15:53 +08:00
}
if (isVip === 'true') {
condition.isVip = true;
}
if (isOper === 'true') {
condition.roles = {$in: ['oper']};
}
if (isAdmin === 'true') {
condition.roles = {$in: ['admin']};
}
if (isOper === 'true' && isAdmin === 'true') {
condition.roles = {$in: ['oper', 'admin']};
}
if (status !== undefined) {
condition.status = status;
}
2017-10-14 23:15:53 +08:00
if (mongoose.Types.ObjectId.isValid(keysA)) {
condition._id = keysA;
} else {
if (keysA.length > 0) {
condition.$or = [
{displayName: {'$all': keysA}},
{email: {'$all': keysA}},
{passkey: {'$all': keysA}},
{username: {'$all': keysA}}
];
}
2017-10-14 23:15:53 +08:00
}
mtDebug.debugBlue(condition, 'ADMIN_USERS_LIST');
2017-10-14 23:15:53 +08:00
var countQuery = function (callback) {
User.count(condition, function (err, count) {
if (err) {
callback(err, null);
} else {
callback(null, count);
}
});
};
var findQuery = function (callback) {
User.find(condition, '-salt -password -providerData')
.sort('-created')
2018-05-13 16:48:15 +08:00
.populate('invited_by', 'username displayName profileImageURL isVip score uploaded downloaded')
.populate('makers', 'name')
2017-10-14 23:15:53 +08:00
.skip(skip)
.limit(limit)
.exec(function (err, users) {
if (err) {
callback(err, null);
} else {
callback(null, users);
}
});
2017-10-14 23:15:53 +08:00
};
2017-10-14 23:15:53 +08:00
async.parallel([countQuery, findQuery], function (err, results) {
if (err) {
mtDebug.debugRed(err);
return res.status(422).send(err);
} else {
res.json({rows: results[1], total: results[0]});
}
});
};
/**
* updateUserRole
* @param req
* @param res
*/
exports.updateUserRole = function (req, res) {
var user = req.model;
user.update({
$set: {roles: req.body.userRole}
}).exec(function (err, result) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
user.roles = req.body.userRole;
res.json(user);
//add server message
if (serverNoticeConfig.action.userRoleChanged.enable) {
serverMessage.addMessage(user._id, serverNoticeConfig.action.userRoleChanged.title, serverNoticeConfig.action.userRoleChanged.content, {
by_name: req.user.displayName,
by_id: req.user._id,
user_role: req.body.userRole[0],
site_name: appConfig.name
});
}
//create trace log
traceLogCreate(req, traceConfig.action.adminUpdateUserRole, {
2017-06-24 13:53:32 +08:00
user: user._id,
role: req.body.userRole[0]
});
//write history
history.insert(user._id, historyConfig.action.adminUpdateUserRole, {
role: req.body.userRole[0],
by: req.user._id
});
}
});
};
/**
* updateUserStatus
* @param req
* @param res
*/
exports.updateUserStatus = function (req, res) {
var user = req.model;
var tp = {
status: req.body.userStatus,
banReason: {
reason: req.body.banReason,
params: undefined
}
};
if (user.status === 'idle' && req.body.userStatus === 'normal') {
tp.last_signed = Date.now();
}
user.update({
$set: tp
}).exec(function (err, result) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
user.status = req.body.userStatus;
res.json(user);
//create trace log
traceLogCreate(req, traceConfig.action.adminUpdateUserStatus, {
2017-06-24 13:53:32 +08:00
user: user._id,
status: req.body.userStatus,
reason: req.body.banReason === '' ? 'VALUE_NULL' : req.body.banReason
});
// write history
history.insert(user._id, historyConfig.action.adminUpdateUserStatus, {
status: req.body.userStatus,
reason: req.body.banReason === '' ? 'VALUE_NULL' : req.body.banReason,
by: req.user._id
});
//ban inviter
if (req.body.userStatus === 'banned' && inviteConfig.banUserInviter && user.invited_by) {
if (!user.invited_by.isOper) {
if (!user.invited_by.isVip || (user.invited_by.isVip && inviteConfig.banUserInviterVip)) {
user.invited_by.update({
$set: {
status: req.body.userStatus,
banReason: {
reason: 'BANNED.YOU_ARE_BANNED_FROM_INVITED_USER',
params: {
uname: user.displayName,
uReason: req.body.banReason
}
}
}
}).exec(function (err, result) {
if (!err) {
//create trace log
traceLogCreate(req, traceConfig.action.adminBanUserInviter, {
user: user.invited_by._id,
status: req.body.userStatus,
invited_user: user._id,
reason: req.body.banReason
});
// write history
history.insert(user.invited_by._id, historyConfig.action.adminBanUserInviter, {
status: req.body.userStatus,
uname: user.displayName,
reason: req.body.banReason === '' ? 'VALUE_NULL' : req.body.banReason,
by: req.user._id
});
}
});
}
}
}
}
});
};
/**
* updateUserScore
* @param req
* @param res
*/
exports.updateUserScore = function (req, res) {
var user = req.model;
var sv = parseFloat(req.body.userScore.toFixed(2));
user.score = Math.max(user.score + sv, 0);
user.save(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
//write score detail log
var sl = new ScoreLog({
user: user,
score: sv,
reason: {
event: scoreConfig.action.adminModify.name,
event_str: scoreConfig.action.adminModify.content,
params: {
uid: req.user._id,
uname: req.user.displayName
}
}
});
sl.save(function (err) {
if (err) {
mtDebug.debugError(err);
}
});
//write score days/months log
dataLog.scoreLog(user, sv);
//create trace log
traceLogCreate(req, traceConfig.action.adminUpdateUserScore, {
2017-06-24 13:53:32 +08:00
user: user._id,
score: sv
});
//write history
history.insert(user._id, historyConfig.action.adminUpdateUserScore, {
score: sv,
by: req.user._id
});
});
};
/**
* updateUserUploaded
* @param req
* @param res
*/
exports.updateUserUploaded = function (req, res) {
var user = req.model;
user.uploaded = user.uploaded + parseInt(req.body.userUploaded, 10);
if (user.uploaded < 0) {
user.uploaded = 0;
}
user.save(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
//create trace log
traceLogCreate(req, traceConfig.action.adminUpdateUserUploaded, {
2017-06-24 13:53:32 +08:00
user: user._id,
uploaded: req.body.userUploaded
});
//write history
history.insert(user._id, historyConfig.action.adminUpdateUserUploaded, {
uploaded: req.body.userUploaded,
by: req.user._id
});
});
};
/**
* updateUserDownloaded
* @param req
* @param res
*/
exports.updateUserDownloaded = function (req, res) {
var user = req.model;
user.downloaded = user.downloaded + parseInt(req.body.userDownloaded, 10);
if (user.downloaded < 0) {
user.downloaded = 0;
}
user.save(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
//create trace log
traceLogCreate(req, traceConfig.action.adminUpdateUserDownloaded, {
2017-06-24 13:53:32 +08:00
user: user._id,
downloaded: req.body.userDownloaded
});
//write history
history.insert(user._id, historyConfig.action.adminUpdateUserDownloaded, {
downloaded: req.body.userDownloaded,
by: req.user._id
});
});
};
/**
* addVIPMonths
* @param req
* @param res
*/
exports.addVIPMonths = function (req, res) {
var user = req.model;
var months = parseInt(req.params.months, 10);
if (months > 0) {
2018-04-29 15:05:18 +08:00
mtDebug.debugBlue(user.vip_start_at, 'VIP_START_AT');
mtDebug.debugBlue(user.vip_end_at, 'VIP_END_AT');
var now = moment(Date.now());
var start = moment(user.vip_start_at);
var end = moment(user.vip_end_at);
if (!user.vip_end_at) {
start = now;
end = moment(start).add(months, 'M');
} else if (now > end) {
start = now;
end = moment(start).add(months, 'M');
} else {
end = moment(end).add(months, 'M');
}
2018-04-29 15:05:18 +08:00
mtDebug.debugBlue(start, 'VIP_NEW_START_AT');
mtDebug.debugBlue(end, 'VIP_NEW_END_AT');
user.vip_start_at = start;
user.vip_end_at = end;
user.status = 'normal';
user.hnr_warning = 0;
user.save(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
//clean all H&R record
Complete.remove({
user: user._id
}).exec();
//add server message
if (serverNoticeConfig.action.userVipStatusChanged.enable) {
serverMessage.addMessage(user._id, serverNoticeConfig.action.userVipStatusChanged.title, serverNoticeConfig.action.userVipStatusChanged.content, {
vip_end_at: user.vip_end_at
});
}
//create trace log
traceLogCreate(req, traceConfig.action.adminUpdateUserVIPData, {
user: user._id,
reset: false,
months: months
});
//write history
history.insert(user._id, historyConfig.action.adminUpdateUserVIPData, {
months: months,
by: req.user._id
});
});
} else {
return res.status(422).send({
message: 'PARAMS_MONTH_ERROR'
});
}
};
/**
* uploaderList
* @param req
* @param res
*/
exports.uploaderList = function (req, res) {
var skip = 0;
var limit = 0;
if (req.query.skip !== undefined) {
skip = parseInt(req.query.skip, 10);
}
if (req.query.limit !== undefined) {
limit = parseInt(req.query.limit, 10);
}
var countQuery = function (callback) {
User.count({
uptotal: {$gt: 0}
}, function (err, count) {
if (err) {
callback(err, null);
} else {
callback(null, count);
}
});
};
var findQuery = function (callback) {
User.find({
uptotal: {$gt: 0}
}, '-salt -password -providerData')
.sort('-uptotal')
.populate('makers', 'name')
.skip(skip)
.limit(limit)
.exec(function (err, users) {
if (err) {
callback(err, null);
} else {
callback(null, users);
}
});
};
async.parallel([countQuery, findQuery], function (err, results) {
if (err) {
mtDebug.debugRed(err);
return res.status(422).send(err);
} else {
res.json({rows: results[1], total: results[0]});
}
});
};
/**
* resetVIPData
* @param req
* @param res
*/
exports.resetVIPData = function (req, res) {
var user = req.model;
user.vip_start_at = '';
user.vip_end_at = '';
user.save(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
//create trace log
traceLogCreate(req, traceConfig.action.adminUpdateUserVIPData, {
user: user._id,
reset: true
});
//write history
history.insert(user._id, historyConfig.action.adminUpdateUserVIPData, {
months: 'reset',
by: req.user._id
});
});
};
/**
* presentInvitations
* @param req
* @param res
*/
exports.presentInvitations = function (req, res) {
var user = req.model;
var numbers = req.body.numbers;
var days = req.body.days;
var invitations = [];
for (var i = 0; i < numbers; i++) {
var invitation = new Invitation();
invitation.expiresat = Date.now() + (60 * 60 * 1000 * 24 * days);
invitation.user = user;
invitation.type = 'present';
invitation.isOfficial = false;
invitation.token = user.randomAsciiString(32);
invitations.push(invitation);
}
Invitation.insertMany(invitations, function (err, docs) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
res.json(user);
//create trace log
traceLogCreate(req, traceConfig.action.adminPresentUserInvitations, {
user: user._id,
invitations: docs
});
//write history
history.insert(user._id, historyConfig.action.adminPresentUserInvitations, {
numbers: numbers,
days: days,
by: req.user._id
});
}
});
};
/**
* getUserHistory
* @param req
* @param res
*/
exports.getUserHistory = function (req, res) {
var user = req.model;
User.findById(user._id, 'history')
.populate({
path: 'history.params.by',
model: 'User',
select: 'displayName'
})
.exec(function (err, his) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
res.json(his.history);
}
});
};
/**
* list user seeding torrents
* @param req
* @param res
*/
exports.getUserSeeding = function (req, res) {
Peer.find({
user: req.model._id,
peer_status: PEERSTATE_SEEDER,
last_announce_at: {$gt: Date.now() - announceConfig.announceInterval - announceConfig.announceIdleTime}
}).sort('-peer_uploaded')
.populate({
path: 'torrent',
select: populateStrings.populate_torrent_string,
populate: [
{path: 'user', select: 'displayName profileImageURL'},
{path: 'maker', select: 'name'}
]
})
.exec(function (err, torrents) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
res.json(torrents);
}
});
};
/**
* list user getUserLeeching torrents
* @param req
* @param res
*/
exports.getUserLeeching = function (req, res) {
Peer.find({
user: req.model._id,
peer_status: PEERSTATE_LEECHER,
last_announce_at: {$gt: Date.now() - announceConfig.announceInterval - announceConfig.announceIdleTime}
}).sort('-peer_downloaded')
.populate({
path: 'torrent',
select: populateStrings.populate_torrent_string,
populate: [
{path: 'user', select: 'displayName profileImageURL'},
{path: 'maker', select: 'name'}
]
})
.exec(function (err, torrents) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
res.json(torrents);
}
});
};
/**
* list user warning torrents
* @param req
* @param res
*/
exports.getUserWarning = function (req, res) {
Complete.find({
user: req.model._id,
hnr_warning: true
}).populate({
path: 'torrent',
select: populateStrings.populate_torrent_string,
populate: [
{path: 'user', select: 'displayName profileImageURL'},
{path: 'maker', select: 'name'}
]
}).exec(function (err, complets) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
res.json(complets);
}
});
};
/**
* getUserUploadedTotal
* @param req
* @param res
*/
exports.getUserUploadedTotal = function (req, res) {
Torrent.count({
user: req.user._id
}, function (err, count) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
} else {
res.json({
total: count
});
}
});
};
/**
* resetUserProfileImage
* @param req
* @param res
*/
exports.resetUserProfileImage = function (req, res) {
var user = req.model;
user.profileImageURL = User.schema.path('profileImageURL').defaultValue;
user.save(function (err) {
if (err) {
return res.status(422).send({
message: errorHandler.getErrorMessage(err)
});
}
res.json(user);
});
};
/**
* User middleware
*/
exports.userByID = function (req, res, next, id) {
if (!mongoose.Types.ObjectId.isValid(id)) {
return res.status(400).send({
message: 'SERVER.INVALID_OBJECTID'
});
}
User.findById(id, '-salt -password -providerData -history')
.populate('invited_by', 'username displayName profileImageURL isVip isOper score uploaded downloaded')
.populate('makers', 'user name')
.exec(function (err, user) {
if (err) {
return next(err);
} else if (!user) {
return next(new Error('Failed to load user ' + id));
}
req.model = user;
next();
});
};