2025-12-18 13:50:39 +08:00
|
|
|
|
package admin
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"bytes"
|
2026-02-23 12:45:37 +08:00
|
|
|
|
"context"
|
2025-12-18 13:50:39 +08:00
|
|
|
|
"encoding/csv"
|
2026-03-01 00:41:38 +08:00
|
|
|
|
"errors"
|
2025-12-18 13:50:39 +08:00
|
|
|
|
"fmt"
|
|
|
|
|
|
"strconv"
|
2026-01-09 19:43:19 +08:00
|
|
|
|
"strings"
|
2025-12-18 13:50:39 +08:00
|
|
|
|
|
2025-12-26 21:22:48 +08:00
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/handler/dto"
|
2026-03-01 00:41:38 +08:00
|
|
|
|
infraerrors "github.com/Wei-Shaw/sub2api/internal/pkg/errors"
|
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"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// RedeemHandler handles admin redeem code management
|
|
|
|
|
|
type RedeemHandler struct {
|
2026-03-01 00:41:38 +08:00
|
|
|
|
adminService service.AdminService
|
|
|
|
|
|
redeemService *service.RedeemService
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// NewRedeemHandler creates a new admin redeem handler
|
2026-03-01 00:41:38 +08:00
|
|
|
|
func NewRedeemHandler(adminService service.AdminService, redeemService *service.RedeemService) *RedeemHandler {
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return &RedeemHandler{
|
2026-03-01 00:41:38 +08:00
|
|
|
|
adminService: adminService,
|
|
|
|
|
|
redeemService: redeemService,
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GenerateRedeemCodesRequest represents generate redeem codes request
|
|
|
|
|
|
type GenerateRedeemCodesRequest struct {
|
|
|
|
|
|
Count int `json:"count" binding:"required,min=1,max=100"`
|
2026-01-29 16:29:59 +08:00
|
|
|
|
Type string `json:"type" binding:"required,oneof=balance concurrency subscription invitation"`
|
2025-12-18 13:50:39 +08:00
|
|
|
|
Value float64 `json:"value" binding:"min=0"`
|
2025-12-28 11:45:41 +08:00
|
|
|
|
GroupID *int64 `json:"group_id"` // 订阅类型必填
|
|
|
|
|
|
ValidityDays int `json:"validity_days" binding:"omitempty,max=36500"` // 订阅类型使用,默认30天,最大100年
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-01 00:41:38 +08:00
|
|
|
|
// CreateAndRedeemCodeRequest represents creating a fixed code and redeeming it for a target user.
|
2026-03-13 21:26:46 +08:00
|
|
|
|
// Type 为 omitempty 而非 required 是为了向后兼容旧版调用方(不传 type 时默认 balance)。
|
2026-03-01 00:41:38 +08:00
|
|
|
|
type CreateAndRedeemCodeRequest struct {
|
2026-03-13 21:26:46 +08:00
|
|
|
|
Code string `json:"code" binding:"required,min=3,max=128"`
|
|
|
|
|
|
Type string `json:"type" binding:"omitempty,oneof=balance concurrency subscription invitation"` // 不传时默认 balance(向后兼容)
|
|
|
|
|
|
Value float64 `json:"value" binding:"required,gt=0"`
|
|
|
|
|
|
UserID int64 `json:"user_id" binding:"required,gt=0"`
|
|
|
|
|
|
GroupID *int64 `json:"group_id"` // subscription 类型必填
|
|
|
|
|
|
ValidityDays int `json:"validity_days" binding:"omitempty,max=36500"` // subscription 类型必填,>0
|
|
|
|
|
|
Notes string `json:"notes"`
|
2026-03-01 00:41:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-18 13:50:39 +08:00
|
|
|
|
// List handles listing all redeem codes with pagination
|
|
|
|
|
|
// GET /api/v1/admin/redeem-codes
|
|
|
|
|
|
func (h *RedeemHandler) List(c *gin.Context) {
|
|
|
|
|
|
page, pageSize := response.ParsePagination(c)
|
|
|
|
|
|
codeType := c.Query("type")
|
|
|
|
|
|
status := c.Query("status")
|
|
|
|
|
|
search := c.Query("search")
|
2026-01-09 19:43:19 +08:00
|
|
|
|
// 标准化和验证 search 参数
|
|
|
|
|
|
search = strings.TrimSpace(search)
|
|
|
|
|
|
if len(search) > 100 {
|
|
|
|
|
|
search = search[:100]
|
|
|
|
|
|
}
|
2025-12-18 13:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
codes, total, err := h.adminService.ListRedeemCodes(c.Request.Context(), page, pageSize, codeType, status, search)
|
|
|
|
|
|
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-19 20:09:35 +08:00
|
|
|
|
out := make([]dto.AdminRedeemCode, 0, len(codes))
|
2025-12-26 21:22:48 +08:00
|
|
|
|
for i := range codes {
|
2026-01-19 20:09:35 +08:00
|
|
|
|
out = append(out, *dto.RedeemCodeFromServiceAdmin(&codes[i]))
|
2025-12-26 21:22:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
response.Paginated(c, out, total, page, pageSize)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GetByID handles getting a redeem code by ID
|
|
|
|
|
|
// GET /api/v1/admin/redeem-codes/:id
|
|
|
|
|
|
func (h *RedeemHandler) GetByID(c *gin.Context) {
|
|
|
|
|
|
codeID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid redeem code ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
code, err := h.adminService.GetRedeemCode(c.Request.Context(), codeID)
|
|
|
|
|
|
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-19 20:09:35 +08:00
|
|
|
|
response.Success(c, dto.RedeemCodeFromServiceAdmin(code))
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Generate handles generating new redeem codes
|
|
|
|
|
|
// POST /api/v1/admin/redeem-codes/generate
|
|
|
|
|
|
func (h *RedeemHandler) Generate(c *gin.Context) {
|
|
|
|
|
|
var req GenerateRedeemCodesRequest
|
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-23 12:45:37 +08:00
|
|
|
|
executeAdminIdempotentJSON(c, "admin.redeem_codes.generate", req, service.DefaultWriteIdempotencyTTL(), func(ctx context.Context) (any, error) {
|
|
|
|
|
|
codes, execErr := h.adminService.GenerateRedeemCodes(ctx, &service.GenerateRedeemCodesInput{
|
|
|
|
|
|
Count: req.Count,
|
|
|
|
|
|
Type: req.Type,
|
|
|
|
|
|
Value: req.Value,
|
|
|
|
|
|
GroupID: req.GroupID,
|
|
|
|
|
|
ValidityDays: req.ValidityDays,
|
|
|
|
|
|
})
|
|
|
|
|
|
if execErr != nil {
|
|
|
|
|
|
return nil, execErr
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
out := make([]dto.AdminRedeemCode, 0, len(codes))
|
|
|
|
|
|
for i := range codes {
|
|
|
|
|
|
out = append(out, *dto.RedeemCodeFromServiceAdmin(&codes[i]))
|
|
|
|
|
|
}
|
|
|
|
|
|
return out, nil
|
2025-12-18 13:50:39 +08:00
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-01 00:41:38 +08:00
|
|
|
|
// CreateAndRedeem creates a fixed redeem code and redeems it for a target user in one step.
|
|
|
|
|
|
// POST /api/v1/admin/redeem-codes/create-and-redeem
|
|
|
|
|
|
func (h *RedeemHandler) CreateAndRedeem(c *gin.Context) {
|
|
|
|
|
|
if h.redeemService == nil {
|
|
|
|
|
|
response.InternalError(c, "redeem service not configured")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var req CreateAndRedeemCodeRequest
|
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
req.Code = strings.TrimSpace(req.Code)
|
2026-03-13 21:26:46 +08:00
|
|
|
|
// 向后兼容:旧版调用方(如 Sub2ApiPay)不传 type 字段,默认当作 balance 充值处理。
|
|
|
|
|
|
// 请勿删除此默认值逻辑,否则会导致旧版调用方 400 报错。
|
|
|
|
|
|
if req.Type == "" {
|
|
|
|
|
|
req.Type = "balance"
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if req.Type == "subscription" {
|
|
|
|
|
|
if req.GroupID == nil {
|
|
|
|
|
|
response.BadRequest(c, "group_id is required for subscription type")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
if req.ValidityDays <= 0 {
|
|
|
|
|
|
response.BadRequest(c, "validity_days must be greater than 0 for subscription type")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-03-01 00:41:38 +08:00
|
|
|
|
|
|
|
|
|
|
executeAdminIdempotentJSON(c, "admin.redeem_codes.create_and_redeem", req, service.DefaultWriteIdempotencyTTL(), func(ctx context.Context) (any, error) {
|
|
|
|
|
|
existing, err := h.redeemService.GetByCode(ctx, req.Code)
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
return h.resolveCreateAndRedeemExisting(ctx, existing, req.UserID)
|
|
|
|
|
|
}
|
|
|
|
|
|
if !errors.Is(err, service.ErrRedeemCodeNotFound) {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
createErr := h.redeemService.CreateCode(ctx, &service.RedeemCode{
|
2026-03-13 21:26:46 +08:00
|
|
|
|
Code: req.Code,
|
|
|
|
|
|
Type: req.Type,
|
|
|
|
|
|
Value: req.Value,
|
|
|
|
|
|
Status: service.StatusUnused,
|
|
|
|
|
|
Notes: req.Notes,
|
|
|
|
|
|
GroupID: req.GroupID,
|
|
|
|
|
|
ValidityDays: req.ValidityDays,
|
2026-03-01 00:41:38 +08:00
|
|
|
|
})
|
|
|
|
|
|
if createErr != nil {
|
|
|
|
|
|
// Unique code race: if code now exists, use idempotent semantics by used_by.
|
|
|
|
|
|
existingAfterCreateErr, getErr := h.redeemService.GetByCode(ctx, req.Code)
|
|
|
|
|
|
if getErr == nil {
|
|
|
|
|
|
return h.resolveCreateAndRedeemExisting(ctx, existingAfterCreateErr, req.UserID)
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, createErr
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
redeemed, redeemErr := h.redeemService.Redeem(ctx, req.UserID, req.Code)
|
|
|
|
|
|
if redeemErr != nil {
|
|
|
|
|
|
return nil, redeemErr
|
|
|
|
|
|
}
|
|
|
|
|
|
return gin.H{"redeem_code": dto.RedeemCodeFromServiceAdmin(redeemed)}, nil
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (h *RedeemHandler) resolveCreateAndRedeemExisting(ctx context.Context, existing *service.RedeemCode, userID int64) (any, error) {
|
|
|
|
|
|
if existing == nil {
|
|
|
|
|
|
return nil, infraerrors.Conflict("REDEEM_CODE_CONFLICT", "redeem code conflict")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// If previous run created the code but crashed before redeem, redeem it now.
|
|
|
|
|
|
if existing.CanUse() {
|
|
|
|
|
|
redeemed, err := h.redeemService.Redeem(ctx, userID, existing.Code)
|
|
|
|
|
|
if err == nil {
|
|
|
|
|
|
return gin.H{"redeem_code": dto.RedeemCodeFromServiceAdmin(redeemed)}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
if !errors.Is(err, service.ErrRedeemCodeUsed) {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
latest, getErr := h.redeemService.GetByCode(ctx, existing.Code)
|
|
|
|
|
|
if getErr == nil {
|
|
|
|
|
|
existing = latest
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if existing.UsedBy != nil && *existing.UsedBy == userID {
|
|
|
|
|
|
return gin.H{"redeem_code": dto.RedeemCodeFromServiceAdmin(existing)}, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil, infraerrors.Conflict("REDEEM_CODE_CONFLICT", "redeem code already used by another user")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-18 13:50:39 +08:00
|
|
|
|
// Delete handles deleting a redeem code
|
|
|
|
|
|
// DELETE /api/v1/admin/redeem-codes/:id
|
|
|
|
|
|
func (h *RedeemHandler) Delete(c *gin.Context) {
|
|
|
|
|
|
codeID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid redeem code ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
err = h.adminService.DeleteRedeemCode(c.Request.Context(), codeID)
|
|
|
|
|
|
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": "Redeem code deleted successfully"})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// BatchDelete handles batch deleting redeem codes
|
|
|
|
|
|
// POST /api/v1/admin/redeem-codes/batch-delete
|
|
|
|
|
|
func (h *RedeemHandler) BatchDelete(c *gin.Context) {
|
|
|
|
|
|
var req struct {
|
|
|
|
|
|
IDs []int64 `json:"ids" binding:"required,min=1"`
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid request: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
deleted, err := h.adminService.BatchDeleteRedeemCodes(c.Request.Context(), req.IDs)
|
|
|
|
|
|
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{
|
|
|
|
|
|
"deleted": deleted,
|
|
|
|
|
|
"message": "Redeem codes deleted successfully",
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Expire handles expiring a redeem code
|
|
|
|
|
|
// POST /api/v1/admin/redeem-codes/:id/expire
|
|
|
|
|
|
func (h *RedeemHandler) Expire(c *gin.Context) {
|
|
|
|
|
|
codeID, err := strconv.ParseInt(c.Param("id"), 10, 64)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
response.BadRequest(c, "Invalid redeem code ID")
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
code, err := h.adminService.ExpireRedeemCode(c.Request.Context(), codeID)
|
|
|
|
|
|
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-19 20:09:35 +08:00
|
|
|
|
response.Success(c, dto.RedeemCodeFromServiceAdmin(code))
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GetStats handles getting redeem code statistics
|
|
|
|
|
|
// GET /api/v1/admin/redeem-codes/stats
|
|
|
|
|
|
func (h *RedeemHandler) GetStats(c *gin.Context) {
|
|
|
|
|
|
// Return mock data for now
|
|
|
|
|
|
response.Success(c, gin.H{
|
2025-12-20 15:29:52 +08:00
|
|
|
|
"total_codes": 0,
|
|
|
|
|
|
"active_codes": 0,
|
|
|
|
|
|
"used_codes": 0,
|
|
|
|
|
|
"expired_codes": 0,
|
2025-12-18 13:50:39 +08:00
|
|
|
|
"total_value_distributed": 0.0,
|
|
|
|
|
|
"by_type": gin.H{
|
|
|
|
|
|
"balance": 0,
|
|
|
|
|
|
"concurrency": 0,
|
|
|
|
|
|
"trial": 0,
|
|
|
|
|
|
},
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Export handles exporting redeem codes to CSV
|
|
|
|
|
|
// GET /api/v1/admin/redeem-codes/export
|
|
|
|
|
|
func (h *RedeemHandler) Export(c *gin.Context) {
|
|
|
|
|
|
codeType := c.Query("type")
|
|
|
|
|
|
status := c.Query("status")
|
|
|
|
|
|
|
|
|
|
|
|
// Get all codes without pagination (use large page size)
|
|
|
|
|
|
codes, _, err := h.adminService.ListRedeemCodes(c.Request.Context(), 1, 10000, codeType, status, "")
|
|
|
|
|
|
if err != nil {
|
2025-12-25 20:52:47 +08:00
|
|
|
|
response.ErrorFrom(c, err)
|
2025-12-18 13:50:39 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Create CSV buffer
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
|
|
writer := csv.NewWriter(&buf)
|
|
|
|
|
|
|
|
|
|
|
|
// Write header
|
2026-02-11 16:39:42 +08:00
|
|
|
|
if err := writer.Write([]string{"id", "code", "type", "value", "status", "used_by", "used_by_email", "used_at", "created_at"}); err != nil {
|
2025-12-20 15:29:52 +08:00
|
|
|
|
response.InternalError(c, "Failed to export redeem codes: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
2025-12-18 13:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
// Write data rows
|
|
|
|
|
|
for _, code := range codes {
|
|
|
|
|
|
usedBy := ""
|
|
|
|
|
|
if code.UsedBy != nil {
|
|
|
|
|
|
usedBy = fmt.Sprintf("%d", *code.UsedBy)
|
|
|
|
|
|
}
|
2026-02-11 16:39:42 +08:00
|
|
|
|
usedByEmail := ""
|
|
|
|
|
|
if code.User != nil {
|
|
|
|
|
|
usedByEmail = code.User.Email
|
|
|
|
|
|
}
|
2025-12-18 13:50:39 +08:00
|
|
|
|
usedAt := ""
|
|
|
|
|
|
if code.UsedAt != nil {
|
|
|
|
|
|
usedAt = code.UsedAt.Format("2006-01-02 15:04:05")
|
|
|
|
|
|
}
|
2025-12-20 15:29:52 +08:00
|
|
|
|
if err := writer.Write([]string{
|
2025-12-18 13:50:39 +08:00
|
|
|
|
fmt.Sprintf("%d", code.ID),
|
|
|
|
|
|
code.Code,
|
|
|
|
|
|
code.Type,
|
|
|
|
|
|
fmt.Sprintf("%.2f", code.Value),
|
|
|
|
|
|
code.Status,
|
|
|
|
|
|
usedBy,
|
2026-02-11 16:39:42 +08:00
|
|
|
|
usedByEmail,
|
2025-12-18 13:50:39 +08:00
|
|
|
|
usedAt,
|
|
|
|
|
|
code.CreatedAt.Format("2006-01-02 15:04:05"),
|
2025-12-20 15:29:52 +08:00
|
|
|
|
}); err != nil {
|
|
|
|
|
|
response.InternalError(c, "Failed to export redeem codes: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
2025-12-18 13:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
writer.Flush()
|
2025-12-20 15:29:52 +08:00
|
|
|
|
if err := writer.Error(); err != nil {
|
|
|
|
|
|
response.InternalError(c, "Failed to export redeem codes: "+err.Error())
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
2025-12-18 13:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
c.Header("Content-Type", "text/csv")
|
|
|
|
|
|
c.Header("Content-Disposition", "attachment; filename=redeem_codes.csv")
|
|
|
|
|
|
c.Data(200, "text/csv", buf.Bytes())
|
|
|
|
|
|
}
|