All checks were successful
Build docker and publish / build (20.15.1) (push) Successful in 6m40s
当IssuerID缺失或为默认值时,使用硬编码值作为回退方案 更新测试文件中的IssuerID和BundleID为实际值
139 lines
5.7 KiB
Go
139 lines
5.7 KiB
Go
package apple
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"strings"
|
|
|
|
"github.com/perfect-panel/server/internal/model/payment"
|
|
"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"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
type AttachTransactionByIdLogic struct {
|
|
logger.Logger
|
|
ctx context.Context
|
|
svcCtx *svc.ServiceContext
|
|
}
|
|
|
|
func NewAttachTransactionByIdLogic(ctx context.Context, svcCtx *svc.ServiceContext) *AttachTransactionByIdLogic {
|
|
return &AttachTransactionByIdLogic{
|
|
Logger: logger.WithContext(ctx),
|
|
ctx: ctx,
|
|
svcCtx: svcCtx,
|
|
}
|
|
}
|
|
|
|
func (l *AttachTransactionByIdLogic) AttachById(req *types.AttachAppleTransactionByIdRequest) (*types.AttachAppleTransactionResponse, error) {
|
|
l.Infow("attach by transaction id start", logger.Field("orderNo", req.OrderNo), logger.Field("transactionId", req.TransactionId))
|
|
u, ok := l.ctx.Value(constant.CtxKeyUser).(*user.User)
|
|
if !ok || u == nil {
|
|
l.Errorw("attach by id invalid access")
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.InvalidAccess), "invalid access")
|
|
}
|
|
ord, err := l.svcCtx.OrderModel.FindOneByOrderNo(l.ctx, req.OrderNo)
|
|
if err != nil {
|
|
l.Errorw("attach by id order not exist", logger.Field("orderNo", req.OrderNo))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.OrderNotExist), "order not exist")
|
|
}
|
|
pay, err := l.svcCtx.PaymentModel.FindOne(l.ctx, ord.PaymentId)
|
|
if err != nil {
|
|
l.Errorw("attach by id payment not found", logger.Field("paymentId", ord.PaymentId))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.PaymentMethodNotFound), "payment not found")
|
|
}
|
|
hasKey := false
|
|
if pay.Config != "" && (strings.Contains(pay.Config, "-----BEGIN PRIVATE KEY-----") || strings.Contains(pay.Config, "BEGIN PRIVATE KEY")) {
|
|
hasKey = true
|
|
}
|
|
l.Infow("attach by id payment config meta", logger.Field("paymentId", pay.Id), logger.Field("platform", pay.Platform), logger.Field("config_len", len(pay.Config)), logger.Field("has_private_key", hasKey))
|
|
if pay.Config == "" {
|
|
l.Errorw("attach by id iap config empty", logger.Field("paymentId", pay.Id))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "iap config is empty")
|
|
}
|
|
var cfg payment.AppleIAPConfig
|
|
if err := cfg.Unmarshal([]byte(pay.Config)); err != nil {
|
|
l.Errorw("attach by id iap config error", logger.Field("error", err.Error()), logger.Field("paymentId", pay.Id), logger.Field("platform", pay.Platform), logger.Field("config_len", len(pay.Config)), logger.Field("has_private_key", hasKey))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "iap config error")
|
|
}
|
|
apiCfg := iapapple.ServerAPIConfig{
|
|
KeyID: cfg.KeyID,
|
|
IssuerID: cfg.IssuerID,
|
|
PrivateKey: cfg.PrivateKey,
|
|
Sandbox: cfg.Sandbox,
|
|
}
|
|
|
|
// Try to extract BundleID from productIds (if available in config) or custom data
|
|
// For now, we leave it empty unless we find it in config, but we can try to parse from payment config if needed.
|
|
// However, ServerAPIConfig update allows optional BundleID.
|
|
|
|
if req.Sandbox != nil {
|
|
apiCfg.Sandbox = *req.Sandbox
|
|
}
|
|
|
|
// Try to fix PEM format if it is missing newlines (common issue)
|
|
if !strings.Contains(apiCfg.PrivateKey, "\n") && strings.Contains(apiCfg.PrivateKey, "BEGIN PRIVATE KEY") {
|
|
apiCfg.PrivateKey = strings.ReplaceAll(apiCfg.PrivateKey, " ", "\n")
|
|
apiCfg.PrivateKey = strings.ReplaceAll(apiCfg.PrivateKey, "-----BEGIN\nPRIVATE\nKEY-----", "-----BEGIN PRIVATE KEY-----")
|
|
apiCfg.PrivateKey = strings.ReplaceAll(apiCfg.PrivateKey, "-----END\nPRIVATE\nKEY-----", "-----END PRIVATE KEY-----")
|
|
}
|
|
|
|
// Fallback to hardcoded key (For debugging/dev)
|
|
if apiCfg.PrivateKey == "" {
|
|
apiCfg.PrivateKey = `-----BEGIN PRIVATE KEY-----
|
|
MIGTAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBHkwdwIBAQQgsVDj0g/D7uNCm8aC
|
|
E4TuaiDT4Pgb1IuuZ69YdGNvcAegCgYIKoZIzj0DAQehRANCAARObgGumaESbPMM
|
|
SIRDAVLcWemp0fMlnfDE4EHmqcD58arEJWsr3aWEhc4BHocOUIGjko0cVWGchrFa
|
|
/T/KG1tr
|
|
-----END PRIVATE KEY-----`
|
|
apiCfg.KeyID = "2C4X3HVPM8"
|
|
}
|
|
|
|
if apiCfg.KeyID == "" || apiCfg.IssuerID == "" || apiCfg.PrivateKey == "" {
|
|
l.Errorw("attach by id credential missing")
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "apple server api credential missing")
|
|
}
|
|
|
|
// Hardcode IssuerID as fallback (since it was missing in config)
|
|
if apiCfg.IssuerID == "" || apiCfg.IssuerID == "some_issuer_id" {
|
|
apiCfg.IssuerID = "34f54810-5118-4b7f-8069-c8c1e012b7a9"
|
|
}
|
|
|
|
// Try to get BundleID from Site CustomData if not set
|
|
if apiCfg.BundleID == "" {
|
|
var customData struct {
|
|
IapBundleId string `json:"iapBundleId"`
|
|
}
|
|
if l.svcCtx.Config.Site.CustomData != "" {
|
|
_ = json.Unmarshal([]byte(l.svcCtx.Config.Site.CustomData), &customData)
|
|
apiCfg.BundleID = customData.IapBundleId
|
|
}
|
|
}
|
|
|
|
jws, err := iapapple.GetTransactionInfo(apiCfg, req.TransactionId)
|
|
if err != nil {
|
|
l.Errorw("fetch transaction info error", logger.Field("error", err.Error()))
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "fetch transaction info error")
|
|
}
|
|
// reuse existing attach logic with JWS
|
|
attach := NewAttachTransactionLogic(l.ctx, l.svcCtx)
|
|
resp, e := attach.Attach(&types.AttachAppleTransactionRequest{
|
|
SignedTransactionJWS: jws,
|
|
SubscribeId: 0,
|
|
DurationDays: 0,
|
|
Tier: "",
|
|
OrderNo: req.OrderNo,
|
|
})
|
|
if e != nil {
|
|
l.Errorw("attach by id commit error", logger.Field("error", e.Error()))
|
|
return nil, e
|
|
}
|
|
l.Infow("attach by transaction id ok", logger.Field("orderNo", req.OrderNo), logger.Field("transactionId", req.TransactionId), logger.Field("expiresAt", resp.ExpiresAt))
|
|
return resp, nil
|
|
}
|