298 lines
10 KiB
JavaScript
298 lines
10 KiB
JavaScript
const User = require('../models/User');
|
||
const Conversation = require('../models/Conversation');
|
||
const Message = require('../models/Message');
|
||
const ProfileView = require('../models/ProfileView');
|
||
|
||
/**
|
||
* @desc Получить список всех пользователей
|
||
* @route GET /api/admin/users
|
||
* @access Admin
|
||
*/
|
||
const getAllUsers = async (req, res) => {
|
||
try {
|
||
const { search, page = 1, limit = 20, isActive } = req.query;
|
||
const skip = (page - 1) * limit;
|
||
|
||
// Создание фильтра
|
||
const filter = {};
|
||
|
||
// Добавляем поиск по имени или email, если указан
|
||
if (search) {
|
||
filter.$or = [
|
||
{ name: { $regex: search, $options: 'i' } },
|
||
{ email: { $regex: search, $options: 'i' } }
|
||
];
|
||
}
|
||
|
||
// Фильтр по статусу активности, если указан
|
||
if (isActive !== undefined) {
|
||
filter.isActive = isActive === 'true';
|
||
}
|
||
|
||
// Исключаем админа из результатов выдачи
|
||
filter.isAdmin = { $ne: true };
|
||
|
||
// Получаем пользователей с пагинацией
|
||
const users = await User.find(filter)
|
||
.select('-password')
|
||
.skip(skip)
|
||
.limit(parseInt(limit))
|
||
.sort({ createdAt: -1 });
|
||
|
||
// Получаем общее количество пользователей для пагинации
|
||
const total = await User.countDocuments(filter);
|
||
|
||
res.json({
|
||
users,
|
||
pagination: {
|
||
page: parseInt(page),
|
||
limit: parseInt(limit),
|
||
total,
|
||
pages: Math.ceil(total / limit)
|
||
}
|
||
});
|
||
} catch (error) {
|
||
console.error('Ошибка при получении списка пользователей:', error);
|
||
res.status(500).json({ message: 'Ошибка сервера при получении списка пользователей' });
|
||
}
|
||
};
|
||
|
||
/**
|
||
* @desc Получить детальную информацию о пользователе
|
||
* @route GET /api/admin/users/:id
|
||
* @access Admin
|
||
*/
|
||
const getUserDetails = async (req, res) => {
|
||
try {
|
||
const userId = req.params.id;
|
||
|
||
const user = await User.findById(userId)
|
||
.select('-password')
|
||
.lean();
|
||
|
||
if (!user) {
|
||
return res.status(404).json({ message: 'Пользователь не найден' });
|
||
}
|
||
|
||
// Получаем статистику для пользователя
|
||
const messagesCount = await Message.countDocuments({ sender: userId });
|
||
const conversationsCount = await Conversation.countDocuments({
|
||
participants: userId
|
||
});
|
||
const profileViewsCount = await ProfileView.countDocuments({
|
||
profileOwner: userId
|
||
});
|
||
const matchesCount = user.matches ? user.matches.length : 0;
|
||
const likesGivenCount = user.liked ? user.liked.length : 0;
|
||
|
||
// Добавляем статистику к данным пользователя
|
||
const userWithStats = {
|
||
...user,
|
||
stats: {
|
||
messagesCount,
|
||
conversationsCount,
|
||
profileViewsCount,
|
||
matchesCount,
|
||
likesGivenCount
|
||
}
|
||
};
|
||
|
||
res.json(userWithStats);
|
||
} catch (error) {
|
||
console.error('Ошибка при получении информации о пользователе:', error);
|
||
res.status(500).json({ message: 'Ошибка сервера при получении информации о пользователе' });
|
||
}
|
||
};
|
||
|
||
/**
|
||
* @desc Заблокировать/разблокировать пользователя
|
||
* @route PUT /api/admin/users/:id/toggle-active
|
||
* @access Admin
|
||
*/
|
||
const toggleUserActive = async (req, res) => {
|
||
try {
|
||
const userId = req.params.id;
|
||
const user = await User.findById(userId);
|
||
|
||
if (!user) {
|
||
return res.status(404).json({ message: 'Пользователь не найден' });
|
||
}
|
||
|
||
// Проверка, чтобы нельзя было заблокировать админа
|
||
if (user.isAdmin) {
|
||
return res.status(403).json({ message: 'Невозможно заблокировать администратора' });
|
||
}
|
||
|
||
// Изменяем статус активности на противоположный
|
||
user.isActive = !user.isActive;
|
||
await user.save();
|
||
|
||
res.json({
|
||
message: user.isActive ? 'Пользователь разблокирован' : 'Пользователь заблокирован',
|
||
isActive: user.isActive
|
||
});
|
||
} catch (error) {
|
||
console.error('Ошибка при изменении статуса пользователя:', error);
|
||
res.status(500).json({ message: 'Ошибка сервера при изменении статуса пользователя' });
|
||
}
|
||
};
|
||
|
||
/**
|
||
* @desc Получить статистику приложения
|
||
* @route GET /api/admin/statistics
|
||
* @access Admin
|
||
*/
|
||
const getAppStatistics = async (req, res) => {
|
||
try {
|
||
// Общая статистика пользователей
|
||
const totalUsers = await User.countDocuments({ isAdmin: { $ne: true } });
|
||
const activeUsers = await User.countDocuments({ isActive: true, isAdmin: { $ne: true } });
|
||
const inactiveUsers = await User.countDocuments({ isActive: false, isAdmin: { $ne: true } });
|
||
|
||
// Статистика по полу
|
||
const maleUsers = await User.countDocuments({ gender: 'male' });
|
||
const femaleUsers = await User.countDocuments({ gender: 'female' });
|
||
const otherGenderUsers = await User.countDocuments({ gender: 'other' });
|
||
|
||
// Статистика сообщений и диалогов
|
||
const totalMessages = await Message.countDocuments();
|
||
const totalConversations = await Conversation.countDocuments();
|
||
|
||
// Статистика просмотров профилей
|
||
const totalProfileViews = await ProfileView.countDocuments();
|
||
|
||
// Средние значения
|
||
const messagesPerUser = totalUsers > 0 ? totalMessages / totalUsers : 0;
|
||
const conversationsPerUser = totalUsers > 0 ? totalConversations / totalUsers : 0;
|
||
|
||
// Новые пользователи за последние 30 дней
|
||
const thirtyDaysAgo = new Date();
|
||
thirtyDaysAgo.setDate(thirtyDaysAgo.getDate() - 30);
|
||
|
||
const newUsers30Days = await User.countDocuments({
|
||
createdAt: { $gte: thirtyDaysAgo },
|
||
isAdmin: { $ne: true }
|
||
});
|
||
|
||
res.json({
|
||
users: {
|
||
total: totalUsers,
|
||
active: activeUsers,
|
||
inactive: inactiveUsers,
|
||
newIn30Days: newUsers30Days,
|
||
genderDistribution: {
|
||
male: maleUsers,
|
||
female: femaleUsers,
|
||
other: otherGenderUsers
|
||
}
|
||
},
|
||
activity: {
|
||
totalMessages,
|
||
totalConversations,
|
||
totalProfileViews,
|
||
averages: {
|
||
messagesPerUser: messagesPerUser.toFixed(2),
|
||
conversationsPerUser: conversationsPerUser.toFixed(2)
|
||
}
|
||
}
|
||
});
|
||
} catch (error) {
|
||
console.error('Ошибка при получении статистики приложения:', error);
|
||
res.status(500).json({ message: 'Ошибка сервера при получении статистики приложения' });
|
||
}
|
||
};
|
||
|
||
/**
|
||
* @desc Получить список всех диалогов
|
||
* @route GET /api/admin/conversations
|
||
* @access Admin
|
||
*/
|
||
const getAllConversations = async (req, res) => {
|
||
try {
|
||
const { page = 1, limit = 20, userId } = req.query;
|
||
const skip = (page - 1) * limit;
|
||
|
||
// Создание фильтра
|
||
const filter = {};
|
||
|
||
// Фильтр по пользователю, если указан
|
||
if (userId) {
|
||
filter.participants = userId;
|
||
}
|
||
|
||
// Получаем диалоги с пагинацией и данными участников
|
||
const conversations = await Conversation.find(filter)
|
||
.populate('participants', 'name email photos')
|
||
.populate('lastMessage')
|
||
.skip(skip)
|
||
.limit(parseInt(limit))
|
||
.sort({ updatedAt: -1 });
|
||
|
||
// Получаем общее количество диалогов для пагинации
|
||
const total = await Conversation.countDocuments(filter);
|
||
|
||
res.json({
|
||
conversations,
|
||
pagination: {
|
||
page: parseInt(page),
|
||
limit: parseInt(limit),
|
||
total,
|
||
pages: Math.ceil(total / limit)
|
||
}
|
||
});
|
||
} catch (error) {
|
||
console.error('Ошибка при получении списка диалогов:', error);
|
||
res.status(500).json({ message: 'Ошибка сервера при получении списка диалогов' });
|
||
}
|
||
};
|
||
|
||
/**
|
||
* @desc Получить сообщения диалога
|
||
* @route GET /api/admin/conversations/:id/messages
|
||
* @access Admin
|
||
*/
|
||
const getConversationMessages = async (req, res) => {
|
||
try {
|
||
const conversationId = req.params.id;
|
||
const { page = 1, limit = 50 } = req.query;
|
||
const skip = (page - 1) * limit;
|
||
|
||
// Проверяем существование диалога
|
||
const conversation = await Conversation.findById(conversationId);
|
||
if (!conversation) {
|
||
return res.status(404).json({ message: 'Диалог не найден' });
|
||
}
|
||
|
||
// Получаем сообщения с пагинацией
|
||
const messages = await Message.find({ conversationId })
|
||
.populate('sender', 'name email')
|
||
.skip(skip)
|
||
.limit(parseInt(limit))
|
||
.sort({ createdAt: -1 });
|
||
|
||
// Получаем общее количество сообщений для пагинации
|
||
const total = await Message.countDocuments({ conversationId });
|
||
|
||
res.json({
|
||
messages,
|
||
pagination: {
|
||
page: parseInt(page),
|
||
limit: parseInt(limit),
|
||
total,
|
||
pages: Math.ceil(total / limit)
|
||
}
|
||
});
|
||
} catch (error) {
|
||
console.error('Ошибка при получении сообщений диалога:', error);
|
||
res.status(500).json({ message: 'Ошибка сервера при получении сообщений диалога' });
|
||
}
|
||
};
|
||
|
||
module.exports = {
|
||
getAllUsers,
|
||
getUserDetails,
|
||
toggleUserActive,
|
||
getAppStatistics,
|
||
getAllConversations,
|
||
getConversationMessages
|
||
}; |