All checks were successful
Build docker and publish / build (20.15.1) (push) Successful in 6m36s
添加苹果IAP通知处理功能,包括解析和验证JWS签名、处理交易状态变更 新增订单号字段用于关联订单处理 实现交易记录的创建和更新逻辑 处理订阅状态的变更和过期时间计算
134 lines
4.3 KiB
Go
134 lines
4.3 KiB
Go
package apple
|
|
|
|
import (
|
|
"context"
|
|
"crypto/sha256"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/hibiken/asynq"
|
|
iapmodel "github.com/perfect-panel/server/internal/model/iap/apple"
|
|
"github.com/perfect-panel/server/internal/model/user"
|
|
"github.com/perfect-panel/server/internal/svc"
|
|
"github.com/perfect-panel/server/internal/types"
|
|
"github.com/perfect-panel/server/pkg/constant"
|
|
iapapple "github.com/perfect-panel/server/pkg/iap/apple"
|
|
"github.com/perfect-panel/server/pkg/logger"
|
|
"github.com/perfect-panel/server/pkg/xerr"
|
|
queueType "github.com/perfect-panel/server/queue/types"
|
|
"github.com/pkg/errors"
|
|
"gorm.io/gorm"
|
|
)
|
|
|
|
type AttachTransactionLogic struct {
|
|
logger.Logger
|
|
ctx context.Context
|
|
svcCtx *svc.ServiceContext
|
|
}
|
|
|
|
func NewAttachTransactionLogic(ctx context.Context, svcCtx *svc.ServiceContext) *AttachTransactionLogic {
|
|
return &AttachTransactionLogic{
|
|
Logger: logger.WithContext(ctx),
|
|
ctx: ctx,
|
|
svcCtx: svcCtx,
|
|
}
|
|
}
|
|
|
|
func (l *AttachTransactionLogic) Attach(req *types.AttachAppleTransactionRequest) (*types.AttachAppleTransactionResponse, error) {
|
|
u, ok := l.ctx.Value(constant.CtxKeyUser).(*user.User)
|
|
if !ok || u == nil {
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.InvalidAccess), "invalid access")
|
|
}
|
|
txPayload, err := iapapple.VerifyTransactionJWS(req.SignedTransactionJWS)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "invalid jws")
|
|
}
|
|
// idempotency: check existing transaction by original id
|
|
var existTx *iapmodel.Transaction
|
|
existTx, _ = iapmodel.NewModel(l.svcCtx.DB, l.svcCtx.Redis).FindByOriginalId(l.ctx, txPayload.OriginalTransactionId)
|
|
pm, _ := iapapple.ParseProductMap(l.svcCtx.Config.Site.CustomData)
|
|
m, ok := pm.Items[txPayload.ProductId]
|
|
var duration int64
|
|
var tier string
|
|
var subscribeId int64
|
|
if ok {
|
|
duration = m.DurationDays
|
|
tier = m.Tier
|
|
subscribeId = m.SubscribeId
|
|
} else {
|
|
if req.DurationDays <= 0 || req.SubscribeId <= 0 {
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "unknown product")
|
|
}
|
|
duration = req.DurationDays
|
|
tier = req.Tier
|
|
subscribeId = req.SubscribeId
|
|
}
|
|
exp := iapapple.CalcExpire(txPayload.PurchaseDate, duration)
|
|
sum := sha256.Sum256([]byte(req.SignedTransactionJWS))
|
|
jwsHash := hex.EncodeToString(sum[:])
|
|
iapTx := &iapmodel.Transaction{
|
|
UserId: u.Id,
|
|
OriginalTransactionId: txPayload.OriginalTransactionId,
|
|
TransactionId: txPayload.TransactionId,
|
|
ProductId: txPayload.ProductId,
|
|
PurchaseAt: txPayload.PurchaseDate,
|
|
RevocationAt: txPayload.RevocationDate,
|
|
JWSHash: jwsHash,
|
|
}
|
|
err = l.svcCtx.DB.Transaction(func(tx *gorm.DB) error {
|
|
if existTx == nil || existTx.Id == 0 {
|
|
if e := tx.Model(&iapmodel.Transaction{}).Create(iapTx).Error; e != nil {
|
|
return e
|
|
}
|
|
}
|
|
// insert user_subscribe
|
|
userSub := user.Subscribe{
|
|
UserId: u.Id,
|
|
SubscribeId: subscribeId,
|
|
StartTime: time.Now(),
|
|
ExpireTime: exp,
|
|
Traffic: 0,
|
|
Download: 0,
|
|
Upload: 0,
|
|
Token: fmt.Sprintf("iap:%s", txPayload.OriginalTransactionId),
|
|
UUID: uuid.New().String(),
|
|
Status: 1,
|
|
}
|
|
if e := l.svcCtx.UserModel.InsertSubscribe(l.ctx, &userSub, tx); e != nil {
|
|
return e
|
|
}
|
|
// optional: mark related order as paid and enqueue activation
|
|
if req.OrderNo != "" {
|
|
orderInfo, e := l.svcCtx.OrderModel.FindOneByOrderNo(l.ctx, req.OrderNo)
|
|
if e != nil {
|
|
// do not fail transaction if order not found; just continue
|
|
return nil
|
|
}
|
|
if orderInfo.Status == 1 {
|
|
if e := l.svcCtx.OrderModel.UpdateOrderStatus(l.ctx, req.OrderNo, 2, tx); e != nil {
|
|
return e
|
|
}
|
|
}
|
|
// enqueue activation regardless (idempotent handler downstream)
|
|
payload := queueType.ForthwithActivateOrderPayload{OrderNo: req.OrderNo}
|
|
bytes, _ := json.Marshal(payload)
|
|
task := asynq.NewTask(queueType.ForthwithActivateOrder, bytes)
|
|
if _, e := l.svcCtx.Queue.EnqueueContext(l.ctx, task); e != nil {
|
|
// non-fatal
|
|
l.Errorw("enqueue activate task error", logger.Field("error", e.Error()))
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.DatabaseInsertError), "insert error: %v", err.Error())
|
|
}
|
|
return &types.AttachAppleTransactionResponse{
|
|
ExpiresAt: exp.Unix(),
|
|
Tier: tier,
|
|
}, nil
|
|
}
|