2025-08-30 21:59:52 +08:00
|
|
|
|
package service
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"blazing/cool"
|
2026-03-11 22:51:52 +08:00
|
|
|
|
basemodel "blazing/modules/base/model"
|
2025-12-08 17:03:43 +08:00
|
|
|
|
"blazing/modules/base/service"
|
2026-01-19 18:51:56 +08:00
|
|
|
|
"blazing/modules/player/model"
|
2025-10-13 18:51:41 +08:00
|
|
|
|
"context"
|
2025-10-20 00:23:16 +08:00
|
|
|
|
"fmt"
|
2026-03-10 23:24:33 +08:00
|
|
|
|
|
2026-03-11 22:51:52 +08:00
|
|
|
|
"github.com/gogf/gf/v2/database/gdb"
|
2026-03-11 12:19:13 +08:00
|
|
|
|
"github.com/gogf/gf/v2/frame/g"
|
2026-03-10 23:24:33 +08:00
|
|
|
|
"github.com/gogf/gf/v2/os/gtime"
|
2025-08-30 21:59:52 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
// nextCatchTime 生成下一个唯一捕捉时间戳(原子递增 max_ts)
|
|
|
|
|
|
func (s *PetService) nextCatchTime(m ...*gdb.Model) (uint32, error) {
|
|
|
|
|
|
dbm := cool.DBM(service.NewBaseSysUserService().Model)
|
|
|
|
|
|
if len(m) > 0 {
|
|
|
|
|
|
dbm = m[0]
|
2025-08-31 00:27:07 +08:00
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
sql := fmt.Sprintf(`
|
|
|
|
|
|
UPDATE %s
|
|
|
|
|
|
SET max_ts = CASE
|
|
|
|
|
|
WHEN max_ts < EXTRACT(EPOCH FROM NOW())::INT THEN EXTRACT(EPOCH FROM NOW())::INT
|
|
|
|
|
|
ELSE max_ts + 1
|
|
|
|
|
|
END
|
|
|
|
|
|
WHERE id = ? AND deleted_at IS NULL
|
|
|
|
|
|
RETURNING max_ts;`, service.NewBaseSysUserService().Model.TableName())
|
2025-11-02 18:56:16 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
ret, err := dbm.Raw(sql, s.userid).All()
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return 0, fmt.Errorf("生成捕捉时间失败: %w", err)
|
|
|
|
|
|
}
|
|
|
|
|
|
arr := ret.Array()
|
|
|
|
|
|
if len(arr) == 0 {
|
|
|
|
|
|
return 0, fmt.Errorf("生成捕捉时间失败: 无返回数据")
|
|
|
|
|
|
}
|
|
|
|
|
|
return arr[0].Uint32(), nil
|
|
|
|
|
|
}
|
2025-11-02 18:56:16 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
// setCatchTime 将 Pet.CatchTime 同步到 Pet.Data.CatchTime
|
|
|
|
|
|
func setCatchTime(pet *model.Pet) {
|
|
|
|
|
|
if pet != nil {
|
|
|
|
|
|
pet.Data.CatchTime = pet.CatchTime
|
2025-11-02 18:56:16 +08:00
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
}
|
2025-11-02 18:56:16 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
// 获取精灵信息 0是仓库,1是放生
|
|
|
|
|
|
func (s *PetService) PetInfo(flag int) []model.Pet {
|
|
|
|
|
|
var tt []model.Pet
|
|
|
|
|
|
if err := s.dbm(s.Model).Where("free", flag).Scan(&tt); err != nil {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
for i := range tt {
|
|
|
|
|
|
tt[i].Data.CatchTime = tt[i].CatchTime
|
|
|
|
|
|
}
|
2025-09-23 15:01:52 +00:00
|
|
|
|
return tt
|
2025-12-12 19:10:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) PetCount(flag int) int {
|
2026-02-13 22:57:05 +08:00
|
|
|
|
ret, err := s.dbm(s.Model).Where("free", flag).Count()
|
2025-12-12 19:10:09 +00:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return 0
|
|
|
|
|
|
}
|
|
|
|
|
|
return ret
|
2025-08-31 00:27:07 +08:00
|
|
|
|
}
|
2026-01-31 19:10:36 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) UpdateFree(catchTime, free uint32) bool {
|
|
|
|
|
|
res, err := s.dbm(s.Model).Where("catch_time", catchTime).Data("free", free).Update()
|
2026-03-25 00:47:53 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return false
|
|
|
|
|
|
}
|
2026-03-19 14:50:11 +08:00
|
|
|
|
r, _ := res.RowsAffected()
|
|
|
|
|
|
return r > 0
|
2026-03-10 22:20:36 +08:00
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
|
|
|
|
|
|
func (s *PetService) UpdatePrice(catchTime, price, free uint32) error {
|
|
|
|
|
|
var item model.Pet
|
2026-03-19 14:50:11 +08:00
|
|
|
|
var feeRate float64
|
2026-03-26 04:51:36 +08:00
|
|
|
|
|
2026-03-19 20:54:52 +08:00
|
|
|
|
if free == 2 {
|
2026-03-19 22:25:10 +08:00
|
|
|
|
t, _ := s.dbm(s.Model).Where("free", 2).Count()
|
2026-03-12 19:33:56 +08:00
|
|
|
|
if t > 3 {
|
|
|
|
|
|
return fmt.Errorf("精灵数量已满")
|
|
|
|
|
|
}
|
2026-03-19 14:50:11 +08:00
|
|
|
|
} else {
|
2026-03-26 04:51:36 +08:00
|
|
|
|
if err := s.dbm(s.Model).Where("catch_time", catchTime).Scan(&item); err != nil {
|
|
|
|
|
|
return err
|
2026-03-19 14:50:11 +08:00
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
_, feeRate, _ = item.GetOffShelfFee()
|
|
|
|
|
|
_ = feeRate // feeRate 仅在 non-2 分支使用,在此声明以统一作用域
|
2026-03-12 19:33:56 +08:00
|
|
|
|
}
|
2026-03-19 14:50:11 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
res, _ := s.dbm(s.Model).Where("catch_time", catchTime).Data(g.Map{
|
|
|
|
|
|
"sale_price": price,
|
|
|
|
|
|
"free": free,
|
|
|
|
|
|
}).Update()
|
|
|
|
|
|
affected, _ := res.RowsAffected()
|
|
|
|
|
|
if affected > 0 && free != 2 && feeRate != 0 {
|
|
|
|
|
|
amount := item.CalculateOffShelfAmount(feeRate)
|
|
|
|
|
|
service.NewBaseSysUserService().UpdateFreeGold(s.userid, -int64(amount*100))
|
2026-03-11 12:19:13 +08:00
|
|
|
|
}
|
|
|
|
|
|
return nil
|
2026-03-10 22:20:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) BuyPet(pid uint32) error {
|
|
|
|
|
|
tt := NewPetService(0).PetInfoOneByID(pid)
|
2026-03-14 01:36:06 +08:00
|
|
|
|
if tt == nil {
|
|
|
|
|
|
return fmt.Errorf("没有此精灵")
|
|
|
|
|
|
}
|
|
|
|
|
|
if tt.IsVip != 0 {
|
|
|
|
|
|
return fmt.Errorf("不允许交易")
|
|
|
|
|
|
}
|
2026-03-19 20:54:52 +08:00
|
|
|
|
if tt.Free != 2 {
|
2026-03-14 01:36:06 +08:00
|
|
|
|
return fmt.Errorf("未上架")
|
|
|
|
|
|
}
|
|
|
|
|
|
if tt.SalePrice == 0 {
|
|
|
|
|
|
return fmt.Errorf("未设置价格")
|
|
|
|
|
|
}
|
|
|
|
|
|
if !tt.UpdateTime.AddDate(0, 0, 1).Before(gtime.Now()) {
|
2026-03-17 16:56:55 +08:00
|
|
|
|
return fmt.Errorf("未到购买时间")
|
2026-03-14 01:36:06 +08:00
|
|
|
|
}
|
2026-03-10 23:24:33 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
return g.DB().Transaction(context.TODO(), func(ctx context.Context, tx gdb.TX) error {
|
|
|
|
|
|
useGold := int64(tt.SalePrice)*102 + int64(tt.SaleCount)*5
|
|
|
|
|
|
var user basemodel.BaseSysUser
|
2026-03-10 22:20:36 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
if err := tx.Model(basemodel.BaseSysUser{}).Where("id", s.userid).Fields("free_gold").Scan(&user); err != nil {
|
2026-03-14 01:36:06 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
if user.FreeGold < useGold {
|
2026-03-11 22:51:52 +08:00
|
|
|
|
return fmt.Errorf("余额不足")
|
|
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
|
|
|
|
|
|
if _, err := tx.Model(basemodel.BaseSysUser{}).Where("id", s.userid).Increment("free_gold", -useGold); err != nil {
|
2026-03-14 01:36:06 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
2026-03-14 10:27:26 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
res, err := tx.Model(s.Model).Where("id", tt.ID).Delete()
|
2026-03-11 22:51:52 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
affected, _ := res.RowsAffected()
|
|
|
|
|
|
if affected == 0 {
|
2026-03-14 10:27:26 +08:00
|
|
|
|
return fmt.Errorf("重复订单")
|
|
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
|
|
|
|
|
|
if _, err := tx.Model(basemodel.BaseSysUser{}).Where("id", tt.PlayerID).Increment("free_gold", int64(tt.SalePrice)*98); err != nil {
|
2026-03-11 22:51:52 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
catchTime, err := s.nextCatchTime(tx.Model(service.NewBaseSysUserService().Model))
|
2026-03-11 22:51:52 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
player := model.Pet{
|
|
|
|
|
|
PlayerID: s.userid,
|
|
|
|
|
|
Data: tt.Data,
|
|
|
|
|
|
CatchTime: catchTime,
|
|
|
|
|
|
Free: 0,
|
|
|
|
|
|
SalePrice: tt.SalePrice,
|
|
|
|
|
|
SaleCount: tt.SaleCount + 1,
|
|
|
|
|
|
}
|
2026-03-11 22:51:52 +08:00
|
|
|
|
player.IsVip = cool.Config.ServerInfo.IsVip
|
2026-03-26 04:51:36 +08:00
|
|
|
|
player.Data.OldCatchTime = 0
|
2026-03-11 22:51:52 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
if _, err := tx.Model(s.Model).Insert(player); err != nil {
|
2026-03-11 22:51:52 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
})
|
2025-12-31 01:35:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) Update(t model.PetInfo) bool {
|
2026-03-26 02:35:43 +08:00
|
|
|
|
_, err := s.dbm(s.Model).Where("catch_time", t.CatchTime).Data("data", t).Update()
|
2026-03-26 02:36:13 +08:00
|
|
|
|
return err == nil
|
2025-08-30 21:59:52 +08:00
|
|
|
|
}
|
2025-08-31 00:27:07 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) PetInfoOneByCatchTime(catchTime uint32) *model.Pet {
|
2026-02-07 20:16:44 +08:00
|
|
|
|
var tt *model.Pet
|
2026-03-26 04:51:36 +08:00
|
|
|
|
if err := s.dbm(s.Model).Where("catch_time", catchTime).Scan(&tt); err != nil || tt == nil {
|
2026-01-20 19:36:56 +08:00
|
|
|
|
return nil
|
|
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
setCatchTime(tt)
|
2025-09-23 15:01:52 +00:00
|
|
|
|
return tt
|
2025-08-31 00:27:07 +08:00
|
|
|
|
}
|
2026-03-10 23:05:43 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) PetInfoOneByID(id uint32) *model.Pet {
|
2026-03-10 23:05:43 +08:00
|
|
|
|
var tt *model.Pet
|
2026-03-26 04:51:36 +08:00
|
|
|
|
if err := s.dbm(s.Model).Where("id", id).Scan(&tt); err != nil || tt == nil {
|
2026-03-10 23:05:43 +08:00
|
|
|
|
return nil
|
|
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
setCatchTime(tt)
|
2026-03-10 23:05:43 +08:00
|
|
|
|
return tt
|
|
|
|
|
|
}
|
2025-12-13 22:51:39 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) PetInfoOneOther(userid, catchTime uint32) model.Pet {
|
2026-02-07 20:16:44 +08:00
|
|
|
|
var tt model.Pet
|
2026-03-26 04:51:36 +08:00
|
|
|
|
s.dbm(s.Model).Where("catch_time", catchTime).Scan(&tt)
|
2025-12-13 22:51:39 +08:00
|
|
|
|
tt.Data.CatchTime = tt.CatchTime
|
|
|
|
|
|
return tt
|
|
|
|
|
|
}
|
2025-12-07 01:43:12 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) PetInfoOneUnscoped(catchTime uint32) *model.Pet {
|
2026-02-07 20:16:44 +08:00
|
|
|
|
var tt *model.Pet
|
2026-03-26 04:51:36 +08:00
|
|
|
|
if err := s.dbm(s.Model).Where("catch_time", catchTime).Unscoped().Scan(&tt); err != nil || tt == nil {
|
2026-01-20 04:40:36 +08:00
|
|
|
|
return nil
|
|
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
setCatchTime(tt)
|
2025-12-07 01:43:12 +08:00
|
|
|
|
return tt
|
|
|
|
|
|
}
|
2025-11-26 01:33:48 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) PetDel(catchTime uint32) {
|
|
|
|
|
|
s.dbm(s.Model).Where("catch_time", catchTime).Delete()
|
2025-11-26 01:33:48 +08:00
|
|
|
|
}
|
2026-02-13 22:57:05 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
func (s *PetService) PetLevelAll() []model.Pet {
|
|
|
|
|
|
var tt []model.Pet
|
|
|
|
|
|
s.dbm(s.Model).As("pp").
|
|
|
|
|
|
Wheref("(pp.data->>'Level')::INT > 100").
|
|
|
|
|
|
OrderDesc("(pp.data->>'Level')::INT").
|
|
|
|
|
|
Scan(&tt)
|
2026-02-13 22:57:05 +08:00
|
|
|
|
return tt
|
2026-02-13 06:44:16 +08:00
|
|
|
|
}
|
2025-08-31 00:27:07 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
// PetAdd 精灵真正添加后的捕捉时间才是真正的时间
|
2026-03-26 05:33:40 +08:00
|
|
|
|
func (s *PetService) PetAdd(y *model.PetInfo, saleCount uint32) (uint32, error) {
|
2025-12-31 21:00:29 +08:00
|
|
|
|
if y == nil {
|
2026-03-26 05:33:40 +08:00
|
|
|
|
return 0, nil
|
2025-12-31 21:00:29 +08:00
|
|
|
|
}
|
2025-12-08 17:03:43 +08:00
|
|
|
|
|
2026-03-26 04:51:36 +08:00
|
|
|
|
catchTime, err := s.nextCatchTime()
|
2025-12-08 17:03:43 +08:00
|
|
|
|
if err != nil {
|
2026-03-26 05:33:40 +08:00
|
|
|
|
return 0, err
|
2026-03-26 04:51:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
y.CatchTime = catchTime
|
|
|
|
|
|
|
|
|
|
|
|
player := model.Pet{
|
|
|
|
|
|
PlayerID: s.userid,
|
|
|
|
|
|
Data: *y,
|
|
|
|
|
|
CatchTime: catchTime,
|
|
|
|
|
|
Free: 0,
|
|
|
|
|
|
SaleCount: saleCount,
|
2025-10-17 19:23:38 +00:00
|
|
|
|
}
|
2026-03-26 04:51:36 +08:00
|
|
|
|
player.IsVip = cool.Config.ServerInfo.IsVip
|
|
|
|
|
|
if _, err := cool.DBM(s.Model).Where("player_id", s.userid).Insert(player); err != nil {
|
2026-03-26 05:33:40 +08:00
|
|
|
|
return 0, err
|
2026-03-26 04:51:36 +08:00
|
|
|
|
}
|
2026-03-26 05:33:40 +08:00
|
|
|
|
return catchTime, nil
|
2025-08-31 06:53:42 +00:00
|
|
|
|
}
|
2026-02-13 06:44:16 +08:00
|
|
|
|
|
2025-10-17 19:40:27 +00:00
|
|
|
|
type PetService struct {
|
2025-11-16 20:30:17 +00:00
|
|
|
|
BaseService
|
2025-10-17 19:40:27 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-16 20:30:17 +00:00
|
|
|
|
func NewPetService(userid uint32) *PetService {
|
2025-10-17 19:40:27 +00:00
|
|
|
|
return &PetService{
|
2025-11-16 20:30:17 +00:00
|
|
|
|
BaseService: BaseService{
|
|
|
|
|
|
userid: userid,
|
|
|
|
|
|
Service: &cool.Service{
|
|
|
|
|
|
Model: model.NewPet(),
|
2026-03-14 01:02:36 +08:00
|
|
|
|
ListQueryOp: &cool.QueryOp{
|
2026-03-19 14:50:11 +08:00
|
|
|
|
AddOrderby: g.MapStrStr{"updateTime": "asc"},
|
2026-03-19 20:54:52 +08:00
|
|
|
|
FieldEQ: []string{"player_id", "free"},
|
2026-03-14 01:02:36 +08:00
|
|
|
|
Where: func(ctx context.Context) [][]interface{} {
|
|
|
|
|
|
return [][]interface{}{
|
2026-03-19 20:54:52 +08:00
|
|
|
|
{"free", 0, false},
|
2026-03-14 01:02:36 +08:00
|
|
|
|
{"is_vip", 0, true},
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
2025-11-16 20:30:17 +00:00
|
|
|
|
PageQueryOp: &cool.QueryOp{
|
2026-03-23 15:01:13 +08:00
|
|
|
|
FieldEQ: []string{"player_id", "free"},
|
|
|
|
|
|
DataFieldEQ: []string{"ID"},
|
2025-11-16 20:30:17 +00:00
|
|
|
|
Where: func(ctx context.Context) [][]interface{} {
|
2026-03-10 20:51:48 +08:00
|
|
|
|
return [][]interface{}{
|
2026-03-19 20:54:52 +08:00
|
|
|
|
{"free", 0, false},
|
2026-03-10 20:51:48 +08:00
|
|
|
|
{"is_vip", 0, true},
|
2025-11-16 20:30:17 +00:00
|
|
|
|
}
|
|
|
|
|
|
},
|
2025-10-13 18:51:41 +08:00
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|