feat: 添加版本和构建时间变量 fix: 修正短信队列类型注释错误 style: 清理未使用的代码和测试文件 docs: 更新安装文档中的下载链接 chore: 迁移数据库脚本添加日志和订阅配置
167 lines
5.6 KiB
Go
167 lines
5.6 KiB
Go
package portal
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/perfect-panel/server/internal/model/order"
|
|
|
|
"github.com/perfect-panel/server/pkg/tool"
|
|
|
|
"github.com/perfect-panel/server/internal/config"
|
|
"github.com/perfect-panel/server/internal/svc"
|
|
"github.com/perfect-panel/server/internal/types"
|
|
"github.com/perfect-panel/server/pkg/constant"
|
|
"github.com/perfect-panel/server/pkg/jwt"
|
|
"github.com/perfect-panel/server/pkg/logger"
|
|
"github.com/perfect-panel/server/pkg/uuidx"
|
|
"github.com/perfect-panel/server/pkg/xerr"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
type QueryPurchaseOrderLogic struct {
|
|
logger.Logger
|
|
ctx context.Context
|
|
svcCtx *svc.ServiceContext
|
|
}
|
|
|
|
// NewQueryPurchaseOrderLogic Query Purchase Order
|
|
func NewQueryPurchaseOrderLogic(ctx context.Context, svcCtx *svc.ServiceContext) *QueryPurchaseOrderLogic {
|
|
return &QueryPurchaseOrderLogic{
|
|
Logger: logger.WithContext(ctx),
|
|
ctx: ctx,
|
|
svcCtx: svcCtx,
|
|
}
|
|
}
|
|
|
|
// Centralized error handler for database issues
|
|
func wrapDatabaseError(err error) error {
|
|
return errors.Wrapf(xerr.NewErrCode(xerr.DatabaseQueryError), "Database Query Error: %v", err.Error())
|
|
}
|
|
|
|
func (l *QueryPurchaseOrderLogic) QueryPurchaseOrder(req *types.QueryPurchaseOrderRequest) (resp *types.QueryPurchaseOrderResponse, err error) {
|
|
orderInfo, err := l.svcCtx.OrderModel.FindOneByOrderNo(l.ctx, req.OrderNo)
|
|
if err != nil {
|
|
return nil, wrapDatabaseError(err)
|
|
}
|
|
// Handle temporary orders if applicable
|
|
var token string
|
|
if orderInfo.Status == 2 || orderInfo.Status == 5 {
|
|
if token, err = l.handleTemporaryOrder(orderInfo, req); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
// Fetch subscription and payment information
|
|
subscribeInfo, paymentInfo, err := l.fetchOrderDetails(orderInfo)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &types.QueryPurchaseOrderResponse{
|
|
OrderNo: orderInfo.OrderNo,
|
|
Subscribe: subscribeInfo,
|
|
Quantity: orderInfo.Quantity,
|
|
Price: orderInfo.Price,
|
|
Amount: orderInfo.Amount,
|
|
Discount: orderInfo.Discount,
|
|
Coupon: orderInfo.Coupon,
|
|
CouponDiscount: orderInfo.CouponDiscount,
|
|
FeeAmount: orderInfo.FeeAmount,
|
|
Payment: paymentInfo,
|
|
Status: orderInfo.Status,
|
|
CreatedAt: orderInfo.CreatedAt.UnixMilli(),
|
|
Token: token,
|
|
}, nil
|
|
}
|
|
|
|
// handleTemporaryOrder processes temporary order-related operations
|
|
func (l *QueryPurchaseOrderLogic) handleTemporaryOrder(orderInfo *order.Order, req *types.QueryPurchaseOrderRequest) (string, error) {
|
|
cacheKey := fmt.Sprintf(constant.TempOrderCacheKey, orderInfo.OrderNo)
|
|
cacheValue, err := l.svcCtx.Redis.Get(l.ctx, cacheKey).Result()
|
|
if err != nil {
|
|
l.Errorw("Get TempOrderCacheKey Error", logger.Field("cacheKey", cacheKey), logger.Field("error", err.Error()))
|
|
return "", errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "Get TempOrderCacheKey Error: %v", err.Error())
|
|
}
|
|
|
|
var tempOrder constant.TemporaryOrderInfo
|
|
if err := json.Unmarshal([]byte(cacheValue), &tempOrder); err != nil {
|
|
l.Errorw("JSON Unmarshal Error", logger.Field("error", err.Error()), logger.Field("cacheValue", cacheValue))
|
|
return "", errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "JSON Unmarshal Error: %v", err.Error())
|
|
}
|
|
if tempOrder.OrderNo != orderInfo.OrderNo {
|
|
return "", errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "Order number mismatch")
|
|
}
|
|
|
|
// Validate user and email
|
|
if err = l.validateUserAndEmail(orderInfo, req.AuthType, req.Identifier); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
// Generate session token
|
|
return l.generateSessionToken(orderInfo.UserId)
|
|
}
|
|
|
|
// validateUserAndEmail ensures the user and email are correct
|
|
func (l *QueryPurchaseOrderLogic) validateUserAndEmail(orderInfo *order.Order, platform, openid string) error {
|
|
userInfo, err := l.svcCtx.UserModel.FindOne(l.ctx, orderInfo.UserId)
|
|
if err != nil {
|
|
return wrapDatabaseError(err)
|
|
}
|
|
|
|
authMethod, err := l.svcCtx.UserModel.FindUserAuthMethodByOpenID(l.ctx, platform, openid)
|
|
if err != nil {
|
|
return wrapDatabaseError(err)
|
|
}
|
|
if authMethod.UserId != userInfo.Id {
|
|
return errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "Email verification failed")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// generateSessionToken creates a session token and stores it in Redis
|
|
func (l *QueryPurchaseOrderLogic) generateSessionToken(userId int64) (string, error) {
|
|
sessionId := uuidx.NewUUID().String()
|
|
token, err := jwt.NewJwtToken(
|
|
l.svcCtx.Config.JwtAuth.AccessSecret,
|
|
time.Now().Unix(),
|
|
l.svcCtx.Config.JwtAuth.AccessExpire,
|
|
jwt.WithOption("UserId", userId),
|
|
jwt.WithOption("SessionId", sessionId),
|
|
)
|
|
if err != nil {
|
|
l.Errorw("Token Generation Error", logger.Field("error", err.Error()))
|
|
return "", errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "Token generation error")
|
|
}
|
|
|
|
cacheKey := fmt.Sprintf("%v:%v", config.SessionIdKey, sessionId)
|
|
if err := l.svcCtx.Redis.Set(l.ctx, cacheKey, userId, time.Duration(l.svcCtx.Config.JwtAuth.AccessExpire)*time.Second).Err(); err != nil {
|
|
return "", errors.Wrapf(xerr.NewErrCode(xerr.ERROR), "Session storage error")
|
|
}
|
|
|
|
return token, nil
|
|
}
|
|
|
|
// fetchOrderDetails retrieves subscription and payment details
|
|
func (l *QueryPurchaseOrderLogic) fetchOrderDetails(orderInfo *order.Order) (types.Subscribe, types.PaymentMethod, error) {
|
|
sub, err := l.svcCtx.SubscribeModel.FindOne(l.ctx, orderInfo.SubscribeId)
|
|
if err != nil {
|
|
return types.Subscribe{}, types.PaymentMethod{}, wrapDatabaseError(err)
|
|
}
|
|
|
|
var subscribeInfo types.Subscribe
|
|
tool.DeepCopy(&subscribeInfo, sub)
|
|
|
|
payment, err := l.svcCtx.PaymentModel.FindOne(l.ctx, orderInfo.PaymentId)
|
|
if err != nil {
|
|
return types.Subscribe{}, types.PaymentMethod{}, wrapDatabaseError(err)
|
|
}
|
|
|
|
var paymentInfo types.PaymentMethod
|
|
tool.DeepCopy(&paymentInfo, payment)
|
|
|
|
return subscribeInfo, paymentInfo, nil
|
|
}
|