2025-04-08 17:15:23 +08:00
|
|
|
|
package handlers
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
2025-04-12 18:03:44 +08:00
|
|
|
|
"platform/pkg/u"
|
2025-04-08 17:15:23 +08:00
|
|
|
|
"platform/web/auth"
|
2025-05-07 17:38:27 +08:00
|
|
|
|
"platform/web/core"
|
2025-05-20 17:14:07 +08:00
|
|
|
|
g "platform/web/globals"
|
2025-11-24 18:44:06 +08:00
|
|
|
|
m "platform/web/models"
|
2025-04-08 17:15:23 +08:00
|
|
|
|
q "platform/web/queries"
|
2025-04-17 18:29:44 +08:00
|
|
|
|
s "platform/web/services"
|
2025-04-08 17:15:23 +08:00
|
|
|
|
"time"
|
|
|
|
|
|
|
2025-11-17 18:38:10 +08:00
|
|
|
|
"gorm.io/gen/field"
|
|
|
|
|
|
|
2025-04-08 17:15:23 +08:00
|
|
|
|
"github.com/gofiber/fiber/v2"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2025-12-30 19:24:03 +08:00
|
|
|
|
// PageResourceShort 分页查询当前用户短效套餐
|
|
|
|
|
|
func PageResourceShort(c *fiber.Ctx) error {
|
2025-04-10 17:49:11 +08:00
|
|
|
|
// 检查权限
|
2025-11-17 18:38:10 +08:00
|
|
|
|
authCtx, err := auth.GetAuthCtx(c).PermitUser()
|
2025-04-10 17:49:11 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 解析请求参数
|
2025-12-30 19:24:03 +08:00
|
|
|
|
req := new(PageResourceShortReq)
|
2025-04-10 17:49:11 +08:00
|
|
|
|
if err := c.BodyParser(req); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-16 18:50:55 +08:00
|
|
|
|
// 查询套餐列表
|
2025-05-22 14:55:04 +08:00
|
|
|
|
do := q.Resource.Where(
|
2025-11-17 18:38:10 +08:00
|
|
|
|
q.Resource.UserID.Eq(authCtx.User.ID),
|
2025-11-24 18:44:06 +08:00
|
|
|
|
q.Resource.Type.Eq(int(m.ResourceTypeShort)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
)
|
2025-04-11 17:36:34 +08:00
|
|
|
|
if req.ResourceNo != nil && *req.ResourceNo != "" {
|
2025-04-30 10:43:30 +08:00
|
|
|
|
do.Where(q.Resource.ResourceNo.Eq(*req.ResourceNo))
|
2025-04-11 17:36:34 +08:00
|
|
|
|
}
|
2025-04-10 17:49:11 +08:00
|
|
|
|
if req.Active != nil {
|
2025-04-30 10:43:30 +08:00
|
|
|
|
do.Where(q.Resource.Active.Is(*req.Active))
|
2025-04-10 17:49:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.Type != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.ResourceShort.As(q.Resource.Short.Name()).Type.Eq(*req.Type))
|
2025-04-10 17:49:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.CreateAfter != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.Resource.CreatedAt.Gte(*req.CreateAfter))
|
2025-04-10 17:49:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.CreateBefore != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.Resource.CreatedAt.Lte(*req.CreateBefore))
|
2025-04-10 17:49:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.ExpireAfter != nil {
|
2025-12-10 20:07:33 +08:00
|
|
|
|
do.Where(q.ResourceShort.As(q.Resource.Short.Name()).ExpireAt.Gte(*req.ExpireAfter))
|
2025-04-10 17:49:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.ExpireBefore != nil {
|
2025-12-10 20:07:33 +08:00
|
|
|
|
do.Where(q.ResourceShort.As(q.Resource.Short.Name()).ExpireAt.Lte(*req.ExpireBefore))
|
2025-04-10 17:49:11 +08:00
|
|
|
|
}
|
2026-03-13 17:03:32 +08:00
|
|
|
|
if req.Status != nil {
|
|
|
|
|
|
var short = q.ResourceShort.As(q.Resource.Short.Name())
|
|
|
|
|
|
switch *req.Status {
|
|
|
|
|
|
case 1:
|
|
|
|
|
|
var timeCond = q.Resource.Where(short.Type.Eq(int(m.ResourceModeTime)), short.ExpireAt.Gte(time.Now()))
|
|
|
|
|
|
var quotaCond = q.Resource.Where(short.Type.Eq(int(m.ResourceModeQuota)), short.Quota.GtCol(short.Used))
|
|
|
|
|
|
do.Where(q.Resource.Where(timeCond).Or(quotaCond))
|
|
|
|
|
|
case 2:
|
|
|
|
|
|
var timeCond = q.Resource.Where(short.Type.Eq(int(m.ResourceModeTime)), short.ExpireAt.Lte(time.Now()))
|
|
|
|
|
|
var quotaCond = q.Resource.Where(short.Type.Eq(int(m.ResourceModeQuota)), short.Quota.LteCol(short.Used))
|
|
|
|
|
|
do.Where(q.Resource.Where(timeCond).Or(quotaCond))
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-04-10 17:49:11 +08:00
|
|
|
|
|
2025-11-28 18:35:33 +08:00
|
|
|
|
resource, err := q.Resource.Where(do).
|
2025-05-17 15:54:42 +08:00
|
|
|
|
Joins(q.Resource.Short).
|
2025-04-12 18:03:44 +08:00
|
|
|
|
Order(q.Resource.CreatedAt.Desc()).
|
2025-04-12 11:48:39 +08:00
|
|
|
|
Offset(req.GetOffset()).
|
|
|
|
|
|
Limit(req.GetLimit()).
|
2025-04-14 16:00:04 +08:00
|
|
|
|
Find()
|
2025-04-10 17:49:11 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-11 17:36:34 +08:00
|
|
|
|
var total int64
|
2025-04-12 11:48:39 +08:00
|
|
|
|
if len(resource) < req.GetLimit() {
|
|
|
|
|
|
total = int64(len(resource) + req.GetOffset())
|
2025-04-11 17:36:34 +08:00
|
|
|
|
} else {
|
2025-04-30 18:22:42 +08:00
|
|
|
|
total, err = q.Resource.
|
|
|
|
|
|
Where(do).
|
|
|
|
|
|
Count()
|
2025-04-11 17:36:34 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
2025-04-10 17:49:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-07 17:38:27 +08:00
|
|
|
|
return c.JSON(core.PageResp{
|
2025-04-10 17:49:11 +08:00
|
|
|
|
Total: int(total),
|
2025-04-12 11:48:39 +08:00
|
|
|
|
Page: req.GetPage(),
|
|
|
|
|
|
Size: req.GetSize(),
|
2025-04-11 17:36:34 +08:00
|
|
|
|
List: resource,
|
2025-04-10 17:49:11 +08:00
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-30 19:24:03 +08:00
|
|
|
|
type PageResourceShortReq struct {
|
2025-05-22 14:55:04 +08:00
|
|
|
|
core.PageReq
|
|
|
|
|
|
ResourceNo *string `json:"resource_no"`
|
|
|
|
|
|
Active *bool `json:"active"`
|
|
|
|
|
|
Type *int `json:"type"`
|
|
|
|
|
|
CreateAfter *time.Time `json:"create_after"`
|
|
|
|
|
|
CreateBefore *time.Time `json:"create_before"`
|
|
|
|
|
|
ExpireAfter *time.Time `json:"expire_after"`
|
|
|
|
|
|
ExpireBefore *time.Time `json:"expire_before"`
|
2026-03-13 17:03:32 +08:00
|
|
|
|
Status *int `json:"status"` // 0 - 全部,1 - 有效,2 - 过期
|
2025-05-22 14:55:04 +08:00
|
|
|
|
}
|
2025-04-10 17:49:11 +08:00
|
|
|
|
|
2025-12-30 19:24:03 +08:00
|
|
|
|
// PageResourceLong 分页查询当前用户长效套餐
|
|
|
|
|
|
func PageResourceLong(c *fiber.Ctx) error {
|
2025-05-22 14:55:04 +08:00
|
|
|
|
// 检查权限
|
2025-11-17 18:38:10 +08:00
|
|
|
|
authCtx, err := auth.GetAuthCtx(c).PermitUser()
|
2025-05-22 14:55:04 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 解析请求参数
|
2025-12-30 19:24:03 +08:00
|
|
|
|
req := new(PageResourceLongReq)
|
2025-05-22 14:55:04 +08:00
|
|
|
|
if err := c.BodyParser(req); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 查询套餐列表
|
|
|
|
|
|
do := q.Resource.Where(
|
2025-11-17 18:38:10 +08:00
|
|
|
|
q.Resource.UserID.Eq(authCtx.User.ID),
|
2025-11-24 18:44:06 +08:00
|
|
|
|
q.Resource.Type.Eq(int(m.ResourceTypeLong)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
)
|
|
|
|
|
|
if req.ResourceNo != nil && *req.ResourceNo != "" {
|
|
|
|
|
|
do.Where(q.Resource.ResourceNo.Eq(*req.ResourceNo))
|
|
|
|
|
|
}
|
|
|
|
|
|
if req.Active != nil {
|
|
|
|
|
|
do.Where(q.Resource.Active.Is(*req.Active))
|
|
|
|
|
|
}
|
|
|
|
|
|
if req.Type != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.ResourceLong.As(q.Resource.Long.Name()).Type.Eq(int(*req.Type)))
|
2025-05-22 14:55:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.CreateAfter != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.Resource.CreatedAt.Gte(*req.CreateAfter))
|
2025-05-22 14:55:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.CreateBefore != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.Resource.CreatedAt.Lte(*req.CreateBefore))
|
2025-05-22 14:55:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.ExpireAfter != nil {
|
2025-12-10 20:07:33 +08:00
|
|
|
|
do.Where(q.ResourceLong.As(q.Resource.Long.Name()).ExpireAt.Gte(*req.ExpireAfter))
|
2025-05-22 14:55:04 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.ExpireBefore != nil {
|
2025-12-10 20:07:33 +08:00
|
|
|
|
do.Where(q.ResourceLong.As(q.Resource.Long.Name()).ExpireAt.Lte(*req.ExpireBefore))
|
2025-05-22 14:55:04 +08:00
|
|
|
|
}
|
2026-03-13 17:03:32 +08:00
|
|
|
|
if req.Status != nil {
|
|
|
|
|
|
var long = q.ResourceLong.As(q.Resource.Long.Name())
|
|
|
|
|
|
switch *req.Status {
|
|
|
|
|
|
case 1:
|
|
|
|
|
|
var timeCond = q.Resource.Where(long.Type.Eq(int(m.ResourceModeTime)), long.ExpireAt.Gte(time.Now()))
|
|
|
|
|
|
var quotaCond = q.Resource.Where(long.Type.Eq(int(m.ResourceModeQuota)), long.Quota.GtCol(long.Used))
|
|
|
|
|
|
do.Where(q.Resource.Where(timeCond).Or(quotaCond))
|
|
|
|
|
|
case 2:
|
|
|
|
|
|
var timeCond = q.Resource.Where(long.Type.Eq(int(m.ResourceModeTime)), long.ExpireAt.Lte(time.Now()))
|
|
|
|
|
|
var quotaCond = q.Resource.Where(long.Type.Eq(int(m.ResourceModeQuota)), long.Quota.LteCol(long.Used))
|
|
|
|
|
|
do.Where(q.Resource.Where(timeCond).Or(quotaCond))
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-05-22 14:55:04 +08:00
|
|
|
|
|
2025-12-04 09:45:35 +08:00
|
|
|
|
resource, err := q.Resource.Where(do).
|
2025-05-22 14:55:04 +08:00
|
|
|
|
Joins(q.Resource.Long).
|
|
|
|
|
|
Order(q.Resource.CreatedAt.Desc()).
|
|
|
|
|
|
Offset(req.GetOffset()).
|
|
|
|
|
|
Limit(req.GetLimit()).
|
|
|
|
|
|
Find()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var total int64
|
|
|
|
|
|
if len(resource) < req.GetLimit() {
|
|
|
|
|
|
total = int64(len(resource) + req.GetOffset())
|
|
|
|
|
|
} else {
|
|
|
|
|
|
total, err = q.Resource.
|
|
|
|
|
|
Where(do).
|
|
|
|
|
|
Count()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return c.JSON(core.PageResp{
|
|
|
|
|
|
Total: int(total),
|
|
|
|
|
|
Page: req.GetPage(),
|
|
|
|
|
|
Size: req.GetSize(),
|
|
|
|
|
|
List: resource,
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
2025-04-12 18:03:44 +08:00
|
|
|
|
|
2025-12-30 19:24:03 +08:00
|
|
|
|
type PageResourceLongReq struct {
|
2025-12-19 10:59:04 +08:00
|
|
|
|
core.PageReq
|
|
|
|
|
|
ResourceNo *string `json:"resource_no"`
|
|
|
|
|
|
Active *bool `json:"active"`
|
|
|
|
|
|
Type *int `json:"type"`
|
|
|
|
|
|
CreateAfter *time.Time `json:"create_after"`
|
|
|
|
|
|
CreateBefore *time.Time `json:"create_before"`
|
|
|
|
|
|
ExpireAfter *time.Time `json:"expire_after"`
|
|
|
|
|
|
ExpireBefore *time.Time `json:"expire_before"`
|
2026-03-13 17:03:32 +08:00
|
|
|
|
Status *int `json:"status"` // 0 - 全部,1 - 有效,2 - 过期
|
2025-04-12 18:03:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-30 19:24:03 +08:00
|
|
|
|
// PageResourceShortByAdmin 分页查询全部短效套餐
|
|
|
|
|
|
func PageResourceShortByAdmin(c *fiber.Ctx) error {
|
|
|
|
|
|
_, err := auth.GetAuthCtx(c).PermitAdmin()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
req := new(struct{ core.PageReq })
|
|
|
|
|
|
if err = g.Validator.ParseBody(c, req); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-06 15:33:55 +08:00
|
|
|
|
list, total, err := q.Resource.
|
2025-12-30 19:24:03 +08:00
|
|
|
|
LeftJoin(q.ResourceShort, q.ResourceShort.ResourceID.EqCol(q.Resource.ID)).
|
|
|
|
|
|
Where(q.Resource.Type.Eq(int(m.ResourceTypeShort))).
|
|
|
|
|
|
FindByPage(req.GetOffset(), req.GetLimit())
|
|
|
|
|
|
|
|
|
|
|
|
return c.JSON(core.PageResp{
|
|
|
|
|
|
List: list,
|
|
|
|
|
|
Total: int(total),
|
|
|
|
|
|
Page: req.GetPage(),
|
|
|
|
|
|
Size: req.GetSize(),
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// PageResourceLongByAdmin 分页查询全部短效套餐
|
|
|
|
|
|
func PageResourceLongByAdmin(c *fiber.Ctx) error {
|
|
|
|
|
|
_, err := auth.GetAuthCtx(c).PermitAdmin()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
req := new(struct{ core.PageReq })
|
|
|
|
|
|
if err = g.Validator.ParseBody(c, req); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
list, total, err := q.Resource.
|
|
|
|
|
|
LeftJoin(q.ResourceLong, q.ResourceLong.ResourceID.EqCol(q.Resource.ID)).
|
|
|
|
|
|
Where(q.Resource.Type.Eq(int(m.ResourceTypeLong))).
|
|
|
|
|
|
FindByPage(req.GetOffset(), req.GetLimit())
|
|
|
|
|
|
|
|
|
|
|
|
return c.JSON(core.PageResp{
|
|
|
|
|
|
List: list,
|
|
|
|
|
|
Total: int(total),
|
|
|
|
|
|
Page: req.GetPage(),
|
|
|
|
|
|
Size: req.GetSize(),
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
// AllActiveResource 所有可用套餐
|
2025-05-22 14:55:04 +08:00
|
|
|
|
func AllActiveResource(c *fiber.Ctx) error {
|
2025-04-12 18:03:44 +08:00
|
|
|
|
// 检查权限
|
2025-11-17 18:38:10 +08:00
|
|
|
|
authCtx, err := auth.GetAuthCtx(c).PermitUser()
|
2025-04-12 18:03:44 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-16 18:50:55 +08:00
|
|
|
|
// 查询套餐列表
|
2025-05-22 14:55:04 +08:00
|
|
|
|
var now = time.Now()
|
|
|
|
|
|
|
|
|
|
|
|
var short = q.ResourceShort.As(q.Resource.Short.Name())
|
|
|
|
|
|
var long = q.ResourceLong.As(q.Resource.Long.Name())
|
|
|
|
|
|
resources, err := q.Resource.
|
|
|
|
|
|
Joins(
|
|
|
|
|
|
q.Resource.Short,
|
|
|
|
|
|
q.Resource.Long,
|
|
|
|
|
|
).
|
2025-04-12 18:03:44 +08:00
|
|
|
|
Where(
|
2025-11-17 18:38:10 +08:00
|
|
|
|
q.Resource.UserID.Eq(authCtx.User.ID),
|
2025-04-12 18:03:44 +08:00
|
|
|
|
q.Resource.Active.Is(true),
|
|
|
|
|
|
q.Resource.Where(
|
2025-11-24 18:44:06 +08:00
|
|
|
|
q.Resource.Type.Eq(int(m.ResourceTypeShort)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
q.ResourceShort.As(q.Resource.Short.Name()).Where(
|
2025-11-24 18:44:06 +08:00
|
|
|
|
short.Type.Eq(int(m.ResourceModeTime)),
|
2025-12-10 20:07:33 +08:00
|
|
|
|
short.ExpireAt.Gte(now),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
q.ResourceShort.As(q.Resource.Short.Name()).
|
2025-12-10 20:07:33 +08:00
|
|
|
|
Where(short.LastAt.Lt(u.Today())).
|
|
|
|
|
|
Or(short.Quota.GtCol(short.Daily)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
).Or(
|
2025-11-24 18:44:06 +08:00
|
|
|
|
short.Type.Eq(int(m.ResourceModeQuota)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
short.Quota.GtCol(short.Used),
|
|
|
|
|
|
),
|
2025-04-12 18:03:44 +08:00
|
|
|
|
).Or(
|
2025-11-24 18:44:06 +08:00
|
|
|
|
q.Resource.Type.Eq(int(m.ResourceTypeLong)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
q.ResourceLong.As(q.Resource.Long.Name()).Where(
|
2025-11-24 18:44:06 +08:00
|
|
|
|
long.Type.Eq(int(m.ResourceModeTime)),
|
2025-12-10 20:07:33 +08:00
|
|
|
|
long.ExpireAt.Gte(now),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
q.ResourceLong.As(q.Resource.Long.Name()).
|
2025-12-10 20:07:33 +08:00
|
|
|
|
Where(long.LastAt.Lt(u.Today())).
|
|
|
|
|
|
Or(long.Quota.GtCol(long.Daily)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
).Or(
|
2025-11-24 18:44:06 +08:00
|
|
|
|
long.Type.Eq(int(m.ResourceModeQuota)),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
long.Quota.GtCol(long.Used),
|
|
|
|
|
|
),
|
2025-04-12 18:03:44 +08:00
|
|
|
|
),
|
2025-05-22 14:55:04 +08:00
|
|
|
|
).
|
2025-04-12 18:03:44 +08:00
|
|
|
|
Order(q.Resource.CreatedAt.Desc()).
|
|
|
|
|
|
Find()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return c.JSON(resources)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
type AllResourceReq struct {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
// StatisticResourceFree 统计每日可用
|
2025-06-20 15:17:15 +08:00
|
|
|
|
func StatisticResourceFree(c *fiber.Ctx) error {
|
|
|
|
|
|
// 检查权限
|
2025-11-17 18:38:10 +08:00
|
|
|
|
authCtx, err := auth.GetAuthCtx(c).PermitUser()
|
2025-06-20 15:17:15 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 统计套餐剩余数量
|
2025-11-21 12:59:05 +08:00
|
|
|
|
resources, err := q.Resource.
|
2025-06-20 15:17:15 +08:00
|
|
|
|
Preload(
|
|
|
|
|
|
q.Resource.Short,
|
|
|
|
|
|
q.Resource.Long,
|
|
|
|
|
|
).
|
|
|
|
|
|
Where(
|
2025-11-17 18:38:10 +08:00
|
|
|
|
q.Resource.UserID.Eq(authCtx.User.ID),
|
2025-06-20 15:17:15 +08:00
|
|
|
|
q.Resource.Active.Is(true),
|
|
|
|
|
|
).
|
2025-07-01 18:23:15 +08:00
|
|
|
|
Select(q.Resource.ID, q.Resource.Type).
|
2025-06-20 15:17:15 +08:00
|
|
|
|
Find()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var shortCount, shortQuotaSum, shortDailyFreeSum int
|
|
|
|
|
|
var longCount, longQuotaSum, longDailyFreeSum int
|
|
|
|
|
|
for _, resource := range resources {
|
|
|
|
|
|
switch {
|
|
|
|
|
|
|
|
|
|
|
|
// 短效包量
|
2025-11-24 18:44:06 +08:00
|
|
|
|
case resource.Type == m.ResourceTypeShort && resource.Short.Type == m.ResourceModeQuota:
|
2025-12-10 20:07:33 +08:00
|
|
|
|
if resource.Short.Quota > resource.Short.Used {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
shortCount++
|
2025-12-10 20:07:33 +08:00
|
|
|
|
shortQuotaSum += int(resource.Short.Quota - resource.Short.Used)
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 长效包量
|
2025-11-24 18:44:06 +08:00
|
|
|
|
case resource.Type == m.ResourceTypeLong && resource.Long.Type == m.ResourceModeQuota:
|
2025-12-10 20:07:33 +08:00
|
|
|
|
if resource.Long.Quota > resource.Long.Used {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
longCount++
|
2025-12-10 20:07:33 +08:00
|
|
|
|
longQuotaSum += int(resource.Long.Quota - resource.Long.Used)
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 短效包时
|
2025-11-24 18:44:06 +08:00
|
|
|
|
case resource.Type == m.ResourceTypeShort && resource.Short.Type == m.ResourceModeTime:
|
2025-12-10 20:07:33 +08:00
|
|
|
|
if time.Time(*resource.Short.ExpireAt).After(time.Now()) {
|
|
|
|
|
|
if resource.Short.LastAt == nil || u.IsToday(time.Time(*resource.Short.LastAt)) == false {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
shortCount++
|
2025-12-10 20:07:33 +08:00
|
|
|
|
shortDailyFreeSum += int(resource.Short.Quota)
|
|
|
|
|
|
} else if resource.Short.Quota > resource.Short.Daily {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
shortCount++
|
2025-12-10 20:07:33 +08:00
|
|
|
|
shortDailyFreeSum += int(resource.Short.Quota - resource.Short.Daily)
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 长效包时
|
2025-11-24 18:44:06 +08:00
|
|
|
|
case resource.Type == m.ResourceTypeLong && resource.Long.Type == m.ResourceModeTime:
|
2025-12-10 20:07:33 +08:00
|
|
|
|
if time.Time(*resource.Long.ExpireAt).After(time.Now()) {
|
|
|
|
|
|
if resource.Long.LastAt == nil || u.IsToday(time.Time(*resource.Long.LastAt)) == false {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
longCount++
|
2025-12-10 20:07:33 +08:00
|
|
|
|
longDailyFreeSum += int(resource.Long.Quota)
|
|
|
|
|
|
} else if resource.Long.Quota > resource.Long.Daily {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
longCount++
|
2025-12-10 20:07:33 +08:00
|
|
|
|
longDailyFreeSum += int(resource.Long.Quota - resource.Long.Daily)
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return c.JSON(StatisticPersonalResp{
|
|
|
|
|
|
Short: StatisticShort{
|
|
|
|
|
|
ResourceCount: shortCount,
|
|
|
|
|
|
ResourceQuotaSum: shortQuotaSum,
|
|
|
|
|
|
ResourceDailyFreeSum: shortDailyFreeSum,
|
|
|
|
|
|
},
|
|
|
|
|
|
Long: StatisticLong{
|
|
|
|
|
|
ResourceCount: longCount,
|
|
|
|
|
|
ResourceQuotaSum: longQuotaSum,
|
|
|
|
|
|
ResourceDailyFreeSum: longDailyFreeSum,
|
|
|
|
|
|
},
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
type StatisticPersonalResp struct {
|
|
|
|
|
|
Short StatisticShort `json:"short"`
|
|
|
|
|
|
Long StatisticLong `json:"long"`
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
type StatisticShort struct {
|
|
|
|
|
|
ResourceCount int `json:"resource_count"`
|
|
|
|
|
|
ResourceQuotaSum int `json:"resource_quota_sum"`
|
|
|
|
|
|
ResourceDailyFreeSum int `json:"resource_daily_free_sum"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
type StatisticLong struct {
|
|
|
|
|
|
ResourceCount int `json:"resource_count"`
|
|
|
|
|
|
ResourceQuotaSum int `json:"resource_quota_sum"`
|
|
|
|
|
|
ResourceDailyFreeSum int `json:"resource_daily_free_sum"`
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
// StatisticResourceUsage 统计每日用量
|
2025-06-20 15:17:15 +08:00
|
|
|
|
func StatisticResourceUsage(c *fiber.Ctx) error {
|
|
|
|
|
|
// 检查权限
|
2025-11-17 18:38:10 +08:00
|
|
|
|
authCtx, err := auth.GetAuthCtx(c).PermitUser()
|
2025-06-20 15:17:15 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
2025-04-12 18:03:44 +08:00
|
|
|
|
|
2025-06-20 15:17:15 +08:00
|
|
|
|
// 解析请求参数
|
|
|
|
|
|
var req = new(StatisticResourceUsageReq)
|
2025-12-04 09:45:35 +08:00
|
|
|
|
if err := g.Validator.ParseBody(c, req); err != nil {
|
2025-06-20 15:17:15 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 统计套餐提取数量
|
2025-12-19 15:13:46 +08:00
|
|
|
|
do := q.LogsUserUsage.Where(
|
|
|
|
|
|
q.LogsUserUsage.UserID.Eq(authCtx.User.ID),
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2025-06-20 15:17:15 +08:00
|
|
|
|
if req.TimeAfter != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.LogsUserUsage.Time.Gte(*req.TimeAfter))
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
if req.TimeBefore != nil {
|
2025-11-24 18:44:06 +08:00
|
|
|
|
do.Where(q.LogsUserUsage.Time.Lte(*req.TimeBefore))
|
2025-06-20 15:17:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var data = new(StatisticResourceUsageResp)
|
2025-11-21 12:59:05 +08:00
|
|
|
|
err = q.LogsUserUsage.
|
2025-06-20 15:17:15 +08:00
|
|
|
|
Select(
|
|
|
|
|
|
q.LogsUserUsage.Count_.Sum().As("count"),
|
2025-06-23 18:32:20 +08:00
|
|
|
|
field.NewUnsafeFieldRaw("date_trunc('day', time)").As("date"),
|
2025-06-20 15:17:15 +08:00
|
|
|
|
).
|
2025-07-01 18:23:15 +08:00
|
|
|
|
Where(do).
|
2025-06-20 15:17:15 +08:00
|
|
|
|
Group(
|
2025-12-19 15:13:46 +08:00
|
|
|
|
field.NewField("", "date"),
|
2025-06-20 15:17:15 +08:00
|
|
|
|
).
|
|
|
|
|
|
Order(
|
|
|
|
|
|
field.NewField("", "date").Desc(),
|
|
|
|
|
|
).
|
|
|
|
|
|
Scan(&data)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return c.JSON(data)
|
|
|
|
|
|
}
|
2025-04-08 17:15:23 +08:00
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
type StatisticResourceUsageReq struct {
|
2025-12-19 15:13:46 +08:00
|
|
|
|
TimeAfter *time.Time `json:"time_start"`
|
|
|
|
|
|
TimeBefore *time.Time `json:"time_end"`
|
2025-04-08 17:15:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
type StatisticResourceUsageResp []struct {
|
|
|
|
|
|
Date time.Time `json:"date"`
|
|
|
|
|
|
Count int `json:"count"`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// CreateResource 创建套餐
|
2025-05-20 17:14:07 +08:00
|
|
|
|
func CreateResource(c *fiber.Ctx) error {
|
2025-04-16 18:50:55 +08:00
|
|
|
|
|
|
|
|
|
|
// 检查权限
|
2025-11-17 18:38:10 +08:00
|
|
|
|
authCtx, err := auth.GetAuthCtx(c).PermitUser()
|
2025-04-16 18:50:55 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 解析请求参数
|
2025-05-20 17:14:07 +08:00
|
|
|
|
var req = new(CreateResourceReq)
|
2025-12-04 09:45:35 +08:00
|
|
|
|
if err := g.Validator.ParseBody(c, req); err != nil {
|
2025-04-16 18:50:55 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-20 17:14:07 +08:00
|
|
|
|
// 创建套餐
|
2025-11-17 18:38:10 +08:00
|
|
|
|
err = s.Resource.CreateResourceByBalance(authCtx.User.ID, time.Now(), req.CreateResourceData)
|
2025-04-08 17:15:23 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-09 16:34:41 +08:00
|
|
|
|
return nil
|
2025-04-08 17:15:23 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-19 10:59:04 +08:00
|
|
|
|
type CreateResourceReq struct {
|
|
|
|
|
|
*s.CreateResourceData
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ResourcePrice 套餐价格
|
2025-05-20 17:14:07 +08:00
|
|
|
|
func ResourcePrice(c *fiber.Ctx) error {
|
2025-04-17 18:29:44 +08:00
|
|
|
|
// 检查权限
|
2025-11-17 18:38:10 +08:00
|
|
|
|
_, err := auth.GetAuthCtx(c).PermitSecretClient()
|
2025-04-16 18:50:55 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-17 18:29:44 +08:00
|
|
|
|
// 解析请求参数
|
2025-06-26 09:28:42 +08:00
|
|
|
|
var req = new(CreateResourceReq)
|
2025-12-04 09:45:35 +08:00
|
|
|
|
if err := g.Validator.ParseBody(c, req); err != nil {
|
2025-12-18 14:22:56 +08:00
|
|
|
|
return core.NewBizErr("接口参数解析异常", err)
|
2025-04-16 18:50:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-06-26 09:28:42 +08:00
|
|
|
|
// 获取套餐价格
|
2025-12-10 20:07:33 +08:00
|
|
|
|
amount, err := req.GetAmount()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
2025-12-19 10:59:04 +08:00
|
|
|
|
|
|
|
|
|
|
// 计算折扣
|
|
|
|
|
|
return c.JSON(ResourcePriceResp{
|
|
|
|
|
|
Price: amount.StringFixed(2),
|
|
|
|
|
|
Discounted: 1,
|
|
|
|
|
|
DiscountedPrice: amount.StringFixed(2),
|
2025-05-20 17:14:07 +08:00
|
|
|
|
})
|
2025-04-16 18:50:55 +08:00
|
|
|
|
}
|
2025-12-19 10:59:04 +08:00
|
|
|
|
|
|
|
|
|
|
type ResourcePriceResp struct {
|
|
|
|
|
|
Price string `json:"price"`
|
|
|
|
|
|
Discounted float32 `json:"discounted"`
|
|
|
|
|
|
DiscountedPrice string `json:"discounted_price"`
|
|
|
|
|
|
}
|