All checks were successful
Build docker and publish / build (20.15.1) (push) Successful in 7m16s
217 lines
8.3 KiB
Go
217 lines
8.3 KiB
Go
package iap
|
||
|
||
import (
|
||
"context"
|
||
"encoding/json"
|
||
"strings"
|
||
"time"
|
||
|
||
"github.com/hibiken/asynq"
|
||
"github.com/perfect-panel/server/internal/model/order"
|
||
"github.com/perfect-panel/server/internal/model/payment"
|
||
"github.com/perfect-panel/server/internal/svc"
|
||
iapapple "github.com/perfect-panel/server/pkg/iap/apple"
|
||
"github.com/perfect-panel/server/pkg/logger"
|
||
pkgpayment "github.com/perfect-panel/server/pkg/payment"
|
||
queueTypes "github.com/perfect-panel/server/queue/types"
|
||
)
|
||
|
||
// ReconcileLogic 第二层:每 5 分钟扫描 trade_no 非空的待支付 IAP 订单
|
||
type ReconcileLogic struct {
|
||
svc *svc.ServiceContext
|
||
}
|
||
|
||
func NewReconcileLogic(svc *svc.ServiceContext) *ReconcileLogic {
|
||
return &ReconcileLogic{svc: svc}
|
||
}
|
||
|
||
func (l *ReconcileLogic) ProcessTask(ctx context.Context, _ *asynq.Task) error {
|
||
logger.Infof("[IAPReconcile] start at %s", time.Now().Format("2006-01-02 15:04:05"))
|
||
return l.reconcile(ctx, 5*time.Minute, 48*time.Hour, false)
|
||
}
|
||
|
||
// reconcile 核心对账逻辑,被第二层和第三层共享
|
||
func (l *ReconcileLogic) reconcile(ctx context.Context, minAge, maxAge time.Duration, requireTradeNo bool) error {
|
||
apiCfg, err := l.loadAppleAPIConfig(ctx)
|
||
if err != nil {
|
||
logger.Errorf("[IAPReconcile] load apple config error: %v", err)
|
||
return nil
|
||
}
|
||
if apiCfg == nil {
|
||
logger.Infof("[IAPReconcile] no enabled apple iap payment found, skip")
|
||
return nil
|
||
}
|
||
|
||
orders, err := l.svc.OrderModel.FindPendingIAPOrders(ctx, minAge, maxAge, requireTradeNo)
|
||
if err != nil {
|
||
logger.Errorf("[IAPReconcile] find pending orders error: %v", err)
|
||
return nil
|
||
}
|
||
if len(orders) == 0 {
|
||
return nil
|
||
}
|
||
logger.Infof("[IAPReconcile] found %d pending IAP orders (requireTradeNo=%v)", len(orders), requireTradeNo)
|
||
|
||
for _, ord := range orders {
|
||
if ord.TradeNo != "" {
|
||
// 有 trade_no:直接用 transactionId 查询 Apple Server API
|
||
l.reconcileByTradeNo(ctx, apiCfg, ord)
|
||
} else if ord.AppAccountToken != "" {
|
||
// trade_no 为空但有 app_account_token:兜底查询
|
||
l.reconcileByAppAccountToken(ctx, apiCfg, ord)
|
||
}
|
||
time.Sleep(100 * time.Millisecond) // 避免 Apple API 限频
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// reconcileByTradeNo 通过 trade_no (originalTransactionId) 直接查询 Apple 交易状态
|
||
func (l *ReconcileLogic) reconcileByTradeNo(ctx context.Context, apiCfg *iapapple.ServerAPIConfig, ord *order.Order) {
|
||
jws, e := iapapple.GetTransactionInfo(*apiCfg, ord.TradeNo)
|
||
if e != nil {
|
||
logger.Errorf("[IAPReconcile] GetTransactionInfo error: orderNo=%s tradeNo=%s err=%v",
|
||
ord.OrderNo, ord.TradeNo, e)
|
||
return
|
||
}
|
||
txPayload, e := iapapple.ParseTransactionJWS(jws)
|
||
if e != nil {
|
||
logger.Errorf("[IAPReconcile] ParseTransactionJWS error: orderNo=%s err=%v", ord.OrderNo, e)
|
||
return
|
||
}
|
||
if txPayload.RevocationDate != nil {
|
||
logger.Infof("[IAPReconcile] transaction revoked, closing order: %s", ord.OrderNo)
|
||
if closeErr := l.svc.OrderModel.UpdateOrderStatus(ctx, ord.OrderNo, 3); closeErr != nil {
|
||
logger.Errorf("[IAPReconcile] close revoked order failed: orderNo=%s err=%v", ord.OrderNo, closeErr)
|
||
}
|
||
return
|
||
}
|
||
// 正常已付款交易 → 先将状态改为已支付,再 enqueue 激活
|
||
if statusErr := l.svc.OrderModel.UpdateOrderStatus(ctx, ord.OrderNo, 2); statusErr != nil {
|
||
logger.Errorf("[IAPReconcile] update order status to paid error: orderNo=%s err=%v", ord.OrderNo, statusErr)
|
||
return
|
||
}
|
||
payload, _ := json.Marshal(queueTypes.ForthwithActivateOrderPayload{OrderNo: ord.OrderNo})
|
||
task := asynq.NewTask(queueTypes.ForthwithActivateOrder, payload)
|
||
if _, e = l.svc.Queue.EnqueueContext(ctx, task); e != nil {
|
||
logger.Errorf("[IAPReconcile] enqueue activate error: orderNo=%s err=%v", ord.OrderNo, e)
|
||
} else {
|
||
logger.Infof("[IAPReconcile] enqueued activate: %s", ord.OrderNo)
|
||
}
|
||
}
|
||
|
||
// reconcileByAppAccountToken 兜底:trade_no 为空但有 app_account_token
|
||
// 通过用户历史交易 ID 查询 Apple 交易历史,匹配 appAccountToken 找到对应交易
|
||
func (l *ReconcileLogic) reconcileByAppAccountToken(ctx context.Context, apiCfg *iapapple.ServerAPIConfig, ord *order.Order) {
|
||
// 1. 获取该用户最近的已知交易 ID
|
||
lastTx, err := l.svc.IAPAppleTransactionModel.FindLatestByUserId(ctx, ord.UserId)
|
||
if err != nil || lastTx == nil || lastTx.OriginalTransactionId == "" {
|
||
logger.Infof("[IAPReconcile-Fallback] no historical transaction found for user %d, orderNo=%s, skip",
|
||
ord.UserId, ord.OrderNo)
|
||
return
|
||
}
|
||
|
||
logger.Infof("[IAPReconcile-Fallback] searching Apple history for orderNo=%s appAccountToken=%s using transactionId=%s",
|
||
ord.OrderNo, ord.AppAccountToken, lastTx.OriginalTransactionId)
|
||
|
||
// 2. 查询 Apple 交易历史
|
||
jwsList, err := iapapple.GetTransactionHistory(*apiCfg, lastTx.OriginalTransactionId)
|
||
if err != nil {
|
||
logger.Errorf("[IAPReconcile-Fallback] GetTransactionHistory error: orderNo=%s err=%v", ord.OrderNo, err)
|
||
return
|
||
}
|
||
|
||
// 3. 在交易历史中查找匹配 appAccountToken 的交易
|
||
for _, jws := range jwsList {
|
||
txPayload, e := iapapple.ParseTransactionJWS(jws)
|
||
if e != nil {
|
||
continue
|
||
}
|
||
if txPayload.AppAccountToken != ord.AppAccountToken {
|
||
continue
|
||
}
|
||
// 匹配成功!
|
||
logger.Infof("[IAPReconcile-Fallback] MATCHED! orderNo=%s appAccountToken=%s transactionId=%s",
|
||
ord.OrderNo, ord.AppAccountToken, txPayload.TransactionId)
|
||
|
||
if txPayload.RevocationDate != nil {
|
||
logger.Infof("[IAPReconcile-Fallback] transaction revoked, closing order: %s", ord.OrderNo)
|
||
if closeErr := l.svc.OrderModel.UpdateOrderStatus(ctx, ord.OrderNo, 3); closeErr != nil {
|
||
logger.Errorf("[IAPReconcile-Fallback] close revoked order failed: orderNo=%s err=%v", ord.OrderNo, closeErr)
|
||
}
|
||
return
|
||
}
|
||
|
||
// 更新 trade_no 为找到的 originalTransactionId
|
||
if updateErr := l.svc.DB.Model(&order.Order{}).
|
||
Where("order_no = ?", ord.OrderNo).
|
||
Update("trade_no", txPayload.OriginalTransactionId).Error; updateErr != nil {
|
||
logger.Errorf("[IAPReconcile-Fallback] update trade_no error: orderNo=%s err=%v", ord.OrderNo, updateErr)
|
||
}
|
||
|
||
// 先将订单状态改为已支付(status=2),activateOrderLogic 要求 status=2 才会处理
|
||
if statusErr := l.svc.OrderModel.UpdateOrderStatus(ctx, ord.OrderNo, 2); statusErr != nil {
|
||
logger.Errorf("[IAPReconcile-Fallback] update order status to paid error: orderNo=%s err=%v", ord.OrderNo, statusErr)
|
||
return
|
||
}
|
||
logger.Infof("[IAPReconcile-Fallback] order status updated to paid: %s", ord.OrderNo)
|
||
|
||
// enqueue 激活
|
||
payload, _ := json.Marshal(queueTypes.ForthwithActivateOrderPayload{OrderNo: ord.OrderNo})
|
||
task := asynq.NewTask(queueTypes.ForthwithActivateOrder, payload)
|
||
if _, e := l.svc.Queue.EnqueueContext(ctx, task); e != nil {
|
||
logger.Errorf("[IAPReconcile-Fallback] enqueue activate error: orderNo=%s err=%v", ord.OrderNo, e)
|
||
} else {
|
||
logger.Infof("[IAPReconcile-Fallback] enqueued activate: %s", ord.OrderNo)
|
||
}
|
||
return
|
||
}
|
||
|
||
logger.Infof("[IAPReconcile-Fallback] no matching transaction found for orderNo=%s appAccountToken=%s (checked %d transactions)",
|
||
ord.OrderNo, ord.AppAccountToken, len(jwsList))
|
||
}
|
||
|
||
// loadAppleAPIConfig 从 payment 表读取第一个启用的 Apple IAP 支付方式配置
|
||
func (l *ReconcileLogic) loadAppleAPIConfig(ctx context.Context) (*iapapple.ServerAPIConfig, error) {
|
||
pays, err := l.svc.PaymentModel.FindListByPlatform(ctx, pkgpayment.AppleIAP.String())
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
for _, pay := range pays {
|
||
if pay.Enable == nil || !*pay.Enable || pay.Config == "" {
|
||
continue
|
||
}
|
||
var cfg payment.AppleIAPConfig
|
||
if e := cfg.Unmarshal([]byte(pay.Config)); e != nil {
|
||
continue
|
||
}
|
||
if cfg.KeyID == "" || cfg.IssuerID == "" || cfg.PrivateKey == "" {
|
||
continue
|
||
}
|
||
apiCfg := &iapapple.ServerAPIConfig{
|
||
KeyID: cfg.KeyID,
|
||
IssuerID: cfg.IssuerID,
|
||
PrivateKey: fixPEM(cfg.PrivateKey),
|
||
Sandbox: cfg.Sandbox,
|
||
}
|
||
// 从 site custom data 读取 BundleID
|
||
var customData struct {
|
||
IapBundleId string `json:"iapBundleId"`
|
||
}
|
||
if l.svc.Config.Site.CustomData != "" {
|
||
_ = json.Unmarshal([]byte(l.svc.Config.Site.CustomData), &customData)
|
||
apiCfg.BundleID = customData.IapBundleId
|
||
}
|
||
return apiCfg, nil
|
||
}
|
||
return nil, nil
|
||
}
|
||
|
||
func fixPEM(key string) string {
|
||
if !strings.Contains(key, "\n") && strings.Contains(key, "BEGIN PRIVATE KEY") {
|
||
key = strings.ReplaceAll(key, " ", "\n")
|
||
key = strings.ReplaceAll(key, "-----BEGIN\nPRIVATE\nKEY-----", "-----BEGIN PRIVATE KEY-----")
|
||
key = strings.ReplaceAll(key, "-----END\nPRIVATE\nKEY-----", "-----END PRIVATE KEY-----")
|
||
}
|
||
return key
|
||
}
|