2025-12-18 13:50:39 +08:00
|
|
|
|
package admin
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"strconv"
|
|
|
|
|
|
|
2025-12-26 21:22:48 +08:00
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/handler/dto"
|
2025-12-24 21:07:21 +08:00
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/pkg/response"
|
|
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/service"
|
2025-12-18 13:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// UserHandler handles admin user management
|
|
|
|
|
|
type UserHandler struct {
|
|
|
|
|
|
adminService service.AdminService
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// NewUserHandler creates a new admin user handler
|
|
|
|
|
|
func NewUserHandler(adminService service.AdminService) *UserHandler {
|
|
|
|
|
|
return &UserHandler{
|
|
|
|
|
|
adminService: adminService,
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// CreateUserRequest represents admin create user request
|
|
|
|
|
|
type CreateUserRequest struct {
|
|
|
|
|
|
Email string `json:"email" binding:"required,email"`
|
|
|
|
|
|
Password string `json:"password" binding:"required,min=6"`
|
2025-12-23 11:26:22 +08:00
|
|
|
|
Username string `json:"username"`
|
|
|
|
|
|
Notes string `json:"notes"`
|
2025-12-18 13:50:39 +08:00
|
|
|
|
Balance float64 `json:"balance"`
|
|
|
|
|
|
Concurrency int `json:"concurrency"`
|
|
|
|
|
|
AllowedGroups []int64 `json:"allowed_groups"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// UpdateUserRequest represents admin update user request
|
|
|
|
|
|
// 使用指针类型来区分"未提供"和"设置为0"
|
|
|
|
|
|
type UpdateUserRequest struct {
|
|
|
|
|
|
Email string `json:"email" binding:"omitempty,email"`
|
|
|
|
|
|
Password string `json:"password" binding:"omitempty,min=6"`
|
2025-12-23 11:26:22 +08:00
|
|
|
|
Username *string `json:"username"`
|
|
|
|
|
|
Notes *string `json:"notes"`
|
2025-12-18 13:50:39 +08:00
|
|
|
|
Balance *float64 `json:"balance"`
|
|
|
|
|
|
Concurrency *int `json:"concurrency"`
|
|
|
|
|
|
Status string `json:"status" binding:"omitempty,oneof=active disabled"`
|
|
|
|
|
|
AllowedGroups *[]int64 `json:"allowed_groups"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// UpdateBalanceRequest represents balance update request
|
|
|
|
|
|
type UpdateBalanceRequest struct {
|
2025-12-23 16:29:57 +08:00
|
|
|
|
Balance float64 `json:"balance" binding:"required,gt=0"`
|
2025-12-18 13:50:39 +08:00
|
|
|
|
Operation string `json:"operation" binding:"required,oneof=set add subtract"`
|
2025-12-23 16:29:57 +08:00
|
|
|
|
Notes string `json:"notes"`
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// List handles listing all users with pagination
|
|
|
|
|
|
// GET /api/v1/admin/users
|
2026-01-01 18:59:38 +08:00
|
|
|
|
// Query params:
|
|
|
|
|
|
// - status: filter by user status
|
|
|
|
|
|
// - role: filter by user role
|
|
|
|
|
|
// - search: search in email, username
|
|
|
|
|
|
// - attr[{id}]: filter by custom attribute value, e.g. attr[1]=company
|
2025-12-18 13:50:39 +08:00
|
|
|
|
func (h *UserHandler) List(c *gin.Context) {
|
|
|
|
|
|
page, pageSize := response.ParsePagination(c)
|
|
|
|
|
|
|
2026-01-01 18:59:38 +08:00
|
|
|
|
filters := service.UserListFilters{
|
|
|
|
|
|
Status: c.Query("status"),
|
|
|
|
|
|
Role: c.Query("role"),
|
|
|
|
|
|
Search: c.Query("search"),
|
|
|
|
|
|
Attributes: parseAttributeFilters(c),
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
users, total, err := h.adminService.ListUsers(c.Request.Context(), page, pageSize, filters)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 21:22:48 +08:00
|
|
|
|
out := make([]dto.User, 0, len(users))
|
|
|
|
|
|
for i := range users {
|
|
|
|
|
|
out = append(out, *dto.UserFromService(&users[i]))
|
|
|
|
|
|
}
|
|
|
|
|
|
response.Paginated(c, out, total, page, pageSize)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-01 18:59:38 +08:00
|
|
|
|
// parseAttributeFilters extracts attribute filters from query params
|
|
|
|
|
|
// Format: attr[{attributeID}]=value, e.g. attr[1]=company&attr[2]=developer
|
|
|
|
|
|
func parseAttributeFilters(c *gin.Context) map[int64]string {
|
|
|
|
|
|
result := make(map[int64]string)
|
|
|
|
|
|
|
|
|
|
|
|
// Get all query params and look for attr[*] pattern
|
|
|
|
|
|
for key, values := range c.Request.URL.Query() {
|
|
|
|
|
|
if len(values) == 0 || values[0] == "" {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
// Check if key matches pattern attr[{id}]
|
|
|
|
|
|
if len(key) > 5 && key[:5] == "attr[" && key[len(key)-1] == ']' {
|
|
|
|
|
|
idStr := key[5 : len(key)-1]
|
|
|
|
|
|
id, err := strconv.ParseInt(idStr, 10, 64)
|
|
|
|
|
|
if err == nil && id > 0 {
|
|
|
|
|
|
result[id] = values[0]
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-18 13:50:39 +08:00
|
|
|
|
// GetByID handles getting a user by ID
|
|
|
|
|
|
// GET /api/v1/admin/users/:id
|
|
|
|
|
|
func (h *UserHandler) GetByID(c *gin.Context) {
|
|
|
|
|
|
userID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid user ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
user, err := h.adminService.GetUser(c.Request.Context(), userID)
|
|
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 21:22:48 +08:00
|
|
|
|
response.Success(c, dto.UserFromService(user))
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Create handles creating a new user
|
|
|
|
|
|
// POST /api/v1/admin/users
|
|
|
|
|
|
func (h *UserHandler) Create(c *gin.Context) {
|
|
|
|
|
|
var req CreateUserRequest
|
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
user, err := h.adminService.CreateUser(c.Request.Context(), &service.CreateUserInput{
|
|
|
|
|
|
Email: req.Email,
|
|
|
|
|
|
Password: req.Password,
|
2025-12-23 11:26:22 +08:00
|
|
|
|
Username: req.Username,
|
|
|
|
|
|
Notes: req.Notes,
|
2025-12-18 13:50:39 +08:00
|
|
|
|
Balance: req.Balance,
|
|
|
|
|
|
Concurrency: req.Concurrency,
|
|
|
|
|
|
AllowedGroups: req.AllowedGroups,
|
|
|
|
|
|
})
|
|
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 21:22:48 +08:00
|
|
|
|
response.Success(c, dto.UserFromService(user))
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Update handles updating a user
|
|
|
|
|
|
// PUT /api/v1/admin/users/:id
|
|
|
|
|
|
func (h *UserHandler) Update(c *gin.Context) {
|
|
|
|
|
|
userID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid user ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var req UpdateUserRequest
|
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 使用指针类型直接传递,nil 表示未提供该字段
|
|
|
|
|
|
user, err := h.adminService.UpdateUser(c.Request.Context(), userID, &service.UpdateUserInput{
|
|
|
|
|
|
Email: req.Email,
|
|
|
|
|
|
Password: req.Password,
|
2025-12-23 11:26:22 +08:00
|
|
|
|
Username: req.Username,
|
|
|
|
|
|
Notes: req.Notes,
|
2025-12-18 13:50:39 +08:00
|
|
|
|
Balance: req.Balance,
|
|
|
|
|
|
Concurrency: req.Concurrency,
|
|
|
|
|
|
Status: req.Status,
|
|
|
|
|
|
AllowedGroups: req.AllowedGroups,
|
|
|
|
|
|
})
|
|
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 21:22:48 +08:00
|
|
|
|
response.Success(c, dto.UserFromService(user))
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Delete handles deleting a user
|
|
|
|
|
|
// DELETE /api/v1/admin/users/:id
|
|
|
|
|
|
func (h *UserHandler) Delete(c *gin.Context) {
|
|
|
|
|
|
userID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid user ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
err = h.adminService.DeleteUser(c.Request.Context(), userID)
|
|
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
response.Success(c, gin.H{"message": "User deleted successfully"})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// UpdateBalance handles updating user balance
|
|
|
|
|
|
// POST /api/v1/admin/users/:id/balance
|
|
|
|
|
|
func (h *UserHandler) UpdateBalance(c *gin.Context) {
|
|
|
|
|
|
userID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid user ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var req UpdateBalanceRequest
|
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-23 16:29:57 +08:00
|
|
|
|
user, err := h.adminService.UpdateUserBalance(c.Request.Context(), userID, req.Balance, req.Operation, req.Notes)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-26 21:22:48 +08:00
|
|
|
|
response.Success(c, dto.UserFromService(user))
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GetUserAPIKeys handles getting user's API keys
|
|
|
|
|
|
// GET /api/v1/admin/users/:id/api-keys
|
|
|
|
|
|
func (h *UserHandler) GetUserAPIKeys(c *gin.Context) {
|
|
|
|
|
|
userID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid user ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
page, pageSize := response.ParsePagination(c)
|
|
|
|
|
|
|
|
|
|
|
|
keys, total, err := h.adminService.GetUserAPIKeys(c.Request.Context(), userID, page, pageSize)
|
|
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-03 06:37:08 -08:00
|
|
|
|
out := make([]dto.ApiKey, 0, len(keys))
|
2025-12-26 21:22:48 +08:00
|
|
|
|
for i := range keys {
|
2026-01-03 06:37:08 -08:00
|
|
|
|
out = append(out, *dto.ApiKeyFromService(&keys[i]))
|
2025-12-26 21:22:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
response.Paginated(c, out, total, page, pageSize)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GetUserUsage handles getting user's usage statistics
|
|
|
|
|
|
// GET /api/v1/admin/users/:id/usage
|
|
|
|
|
|
func (h *UserHandler) GetUserUsage(c *gin.Context) {
|
|
|
|
|
|
userID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid user ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
period := c.DefaultQuery("period", "month")
|
|
|
|
|
|
|
|
|
|
|
|
stats, err := h.adminService.GetUserUsageStats(c.Request.Context(), userID, period)
|
|
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
response.Success(c, stats)
|
|
|
|
|
|
}
|