server/internal/logic/public/portal/queryPurchaseOrderLogic.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
}