Files
juwan-backend/app/wallet/api/internal/logic/wallet/topupLogic.go
T

126 lines
3.1 KiB
Go

// Code scaffolded by goctl. Safe to edit.
// goctl 1.9.2
package wallet
import (
"context"
"errors"
"fmt"
"strings"
"time"
"juwan-backend/app/wallet/rpc/pb"
"juwan-backend/common/utils/contextj"
"juwan-backend/app/wallet/api/internal/svc"
"juwan-backend/app/wallet/api/internal/types"
"github.com/shopspring/decimal"
"github.com/zeromicro/go-zero/core/logx"
)
type TopupLogic struct {
logx.Logger
ctx context.Context
svcCtx *svc.ServiceContext
}
// 充值
func NewTopupLogic(ctx context.Context, svcCtx *svc.ServiceContext) *TopupLogic {
return &TopupLogic{
Logger: logx.WithContext(ctx),
ctx: ctx,
svcCtx: svcCtx,
}
}
func (l *TopupLogic) Topup(req *types.TopupReq) (resp *types.EmptyResp, err error) {
amount, err := decimal.NewFromString(req.Amount)
if err != nil {
return nil, errors.New("invalid amount")
}
if !amount.GreaterThan(decimal.Zero) {
return nil, errors.New("amount must be greater than 0")
}
userID, err := contextj.UserIDFrom(l.ctx)
if err != nil {
return nil, err
}
walletResp, err := l.svcCtx.WalletRpc.GetWalletsById(l.ctx, &pb.GetWalletsByIdReq{Id: userID})
if err != nil {
return nil, err
}
if walletResp.GetWallets() == nil {
zeroAmount := "0"
_, err = l.svcCtx.WalletRpc.AddWallets(l.ctx, &pb.AddWalletsReq{
UserId: userID,
Balance: zeroAmount,
FrozenBalance: zeroAmount,
UpdatedAt: time.Now().Unix(),
})
if err != nil {
return nil, err
}
walletResp, err = l.svcCtx.WalletRpc.GetWalletsById(l.ctx, &pb.GetWalletsByIdReq{Id: userID})
if err != nil {
return nil, err
}
if walletResp.GetWallets() == nil {
return nil, errors.New("wallet not found")
}
}
const maxRetries = 3
for i := 0; i < maxRetries; i++ {
currentBalance, perr := decimal.NewFromString(walletResp.Wallets.Balance)
if perr != nil {
return nil, errors.New("invalid wallet balance")
}
frozenBalance, perr := decimal.NewFromString(walletResp.Wallets.FrozenBalance)
if perr != nil {
return nil, errors.New("invalid wallet frozen balance")
}
expectedVersion := walletResp.Wallets.Version
newBalance := currentBalance.Add(amount)
newBalanceStr := newBalance.String()
frozenBalanceStr := frozenBalance.String()
updatedAt := time.Now().Unix()
_, uerr := l.svcCtx.WalletRpc.UpdateWallets(l.ctx, &pb.UpdateWalletsReq{
UserId: userID,
Balance: &newBalanceStr,
FrozenBalance: &frozenBalanceStr,
UpdatedAt: &updatedAt,
Version: &expectedVersion,
})
if uerr == nil {
desc := fmt.Sprintf("topup via %s", req.Method)
_, err = l.svcCtx.WalletRpc.AddWalletTransactions(l.ctx, &pb.AddWalletTransactionsReq{
UserId: userID,
Type: "topup",
Amount: amount.String(),
BalanceAfter: newBalanceStr,
Description: desc,
CreatedAt: updatedAt,
})
if err != nil {
return nil, err
}
return &types.EmptyResp{}, nil
}
if !strings.Contains(uerr.Error(), "conflict") {
return nil, uerr
}
walletResp, err = l.svcCtx.WalletRpc.GetWalletsById(l.ctx, &pb.GetWalletsByIdReq{Id: userID})
if err != nil {
return nil, err
}
}
return nil, errors.New("wallet update conflict, please retry")
}