- AdminPlansController & Service (CRUD, reorder) - AdminPaymentMethodsController & Service (CRUD, reorder) - AdminPaymentsController & Service (verify, reject, pending count) - AdminUsersController & Service (search, suspend, grant pro access, stats) - AdminConfigController & Service (dynamic config management) - Wire all controllers into AdminModule - Import AdminModule in AppModule Admin API Routes: - GET/POST/PUT/DELETE /admin/plans - GET/POST/PUT/DELETE /admin/payment-methods - GET /admin/payments (with status filter) - POST /admin/payments/:id/verify - POST /admin/payments/:id/reject - GET /admin/users (with search) - POST /admin/users/:id/grant-pro - GET/POST/DELETE /admin/config All routes protected by AuthGuard + AdminGuard
144 lines
3.2 KiB
TypeScript
144 lines
3.2 KiB
TypeScript
import { Injectable } from '@nestjs/common';
|
|
import { PrismaService } from '../prisma/prisma.service';
|
|
|
|
@Injectable()
|
|
export class AdminUsersService {
|
|
constructor(private readonly prisma: PrismaService) {}
|
|
|
|
async findAll(search?: string) {
|
|
return this.prisma.user.findMany({
|
|
where: search
|
|
? {
|
|
OR: [
|
|
{ email: { contains: search, mode: 'insensitive' } },
|
|
{ name: { contains: search, mode: 'insensitive' } },
|
|
],
|
|
}
|
|
: undefined,
|
|
select: {
|
|
id: true,
|
|
email: true,
|
|
name: true,
|
|
role: true,
|
|
emailVerified: true,
|
|
createdAt: true,
|
|
lastLoginAt: true,
|
|
suspendedAt: true,
|
|
_count: {
|
|
select: {
|
|
wallets: true,
|
|
transactions: true,
|
|
},
|
|
},
|
|
},
|
|
orderBy: { createdAt: 'desc' },
|
|
});
|
|
}
|
|
|
|
async findOne(id: string) {
|
|
return this.prisma.user.findUnique({
|
|
where: { id },
|
|
include: {
|
|
subscriptions: {
|
|
include: {
|
|
plan: true,
|
|
},
|
|
},
|
|
_count: {
|
|
select: {
|
|
wallets: true,
|
|
transactions: true,
|
|
payments: true,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
}
|
|
|
|
async updateRole(id: string, role: string) {
|
|
return this.prisma.user.update({
|
|
where: { id },
|
|
data: { role },
|
|
});
|
|
}
|
|
|
|
async suspend(id: string, reason: string) {
|
|
return this.prisma.user.update({
|
|
where: { id },
|
|
data: {
|
|
suspendedAt: new Date(),
|
|
suspendedReason: reason,
|
|
},
|
|
});
|
|
}
|
|
|
|
async unsuspend(id: string) {
|
|
return this.prisma.user.update({
|
|
where: { id },
|
|
data: {
|
|
suspendedAt: null,
|
|
suspendedReason: null,
|
|
},
|
|
});
|
|
}
|
|
|
|
async grantProAccess(userId: string, planSlug: string, durationDays: number) {
|
|
const plan = await this.prisma.plan.findUnique({
|
|
where: { slug: planSlug },
|
|
});
|
|
|
|
if (!plan) {
|
|
throw new Error('Plan not found');
|
|
}
|
|
|
|
const now = new Date();
|
|
const endDate = new Date(now);
|
|
endDate.setDate(endDate.getDate() + durationDays);
|
|
|
|
// Check if user already has a subscription
|
|
const existing = await this.prisma.subscription.findUnique({
|
|
where: { userId },
|
|
});
|
|
|
|
if (existing) {
|
|
// Update existing subscription
|
|
return this.prisma.subscription.update({
|
|
where: { userId },
|
|
data: {
|
|
planId: plan.id,
|
|
status: 'active',
|
|
startDate: now,
|
|
endDate,
|
|
},
|
|
});
|
|
} else {
|
|
// Create new subscription
|
|
return this.prisma.subscription.create({
|
|
data: {
|
|
userId,
|
|
planId: plan.id,
|
|
status: 'active',
|
|
startDate: now,
|
|
endDate,
|
|
},
|
|
});
|
|
}
|
|
}
|
|
|
|
async getStats() {
|
|
const totalUsers = await this.prisma.user.count();
|
|
const activeSubscriptions = await this.prisma.subscription.count({
|
|
where: { status: 'active' },
|
|
});
|
|
const suspendedUsers = await this.prisma.user.count({
|
|
where: { suspendedAt: { not: null } },
|
|
});
|
|
|
|
return {
|
|
totalUsers,
|
|
activeSubscriptions,
|
|
suspendedUsers,
|
|
};
|
|
}
|
|
}
|