fusenapi/service/repositories/order.go

1861 lines
64 KiB
Go
Raw Normal View History

2023-09-15 09:58:45 +00:00
package repositories
import (
"context"
"encoding/json"
"errors"
2023-09-21 11:01:48 +00:00
"fmt"
2023-09-19 11:17:04 +00:00
"fusenapi/constants"
2023-09-15 09:58:45 +00:00
"fusenapi/model/gmodel"
"fusenapi/utils/basic"
"fusenapi/utils/file"
2023-09-21 11:01:48 +00:00
"fusenapi/utils/handlers"
"fusenapi/utils/hash"
2023-09-18 07:39:12 +00:00
"fusenapi/utils/order"
2023-09-21 11:01:48 +00:00
"fusenapi/utils/pay"
"fusenapi/utils/pdf"
2023-09-28 09:53:41 +00:00
"fusenapi/utils/queue"
2023-10-07 08:12:29 +00:00
"math"
2023-09-19 11:17:04 +00:00
"time"
2023-09-15 09:58:45 +00:00
"github.com/aws/aws-sdk-go/aws/session"
2023-09-25 09:31:42 +00:00
"github.com/stripe/stripe-go/v75"
2023-09-22 08:27:17 +00:00
"github.com/zeromicro/go-zero/core/logc"
2023-09-15 09:58:45 +00:00
"gorm.io/gorm"
)
2023-10-10 09:17:28 +00:00
func NewOrder(gormDB *gorm.DB, awsSession *session.Session, delayQueue *queue.DelayMessage) Order {
2023-09-15 09:58:45 +00:00
return &defaultOrder{
2023-09-28 09:53:41 +00:00
MysqlConn: gormDB,
DelayQueue: delayQueue,
AwsSession: awsSession,
2023-09-15 09:58:45 +00:00
}
}
type (
defaultOrder struct {
2023-09-28 09:53:41 +00:00
MysqlConn *gorm.DB
DelayQueue *queue.DelayMessage
AwsSession *session.Session
2023-09-15 09:58:45 +00:00
}
Order interface {
// 下单
2023-09-20 07:07:12 +00:00
Create(ctx context.Context, in *CreateReq) (res *CreateRes, err error)
2023-09-21 11:01:48 +00:00
// 预支付--定金
CreatePrePaymentByDeposit(ctx context.Context, in *CreatePrePaymentByDepositReq) (res *CreatePrePaymentByDepositRes, err error)
2023-09-22 02:39:02 +00:00
// 预支付--定金
CreatePrePaymentByBalance(ctx context.Context, in *CreatePrePaymentByBalanceReq) (res *CreatePrePaymentByBalanceRes, err error)
2023-09-15 09:58:45 +00:00
// 列表
2023-09-21 11:01:48 +00:00
List(ctx context.Context, in *ListReq) (res *ListRes, err error)
2023-09-15 09:58:45 +00:00
// 详情
2023-09-21 02:30:36 +00:00
Detail(ctx context.Context, in *DetailReq) (res *DetailRes, err error)
// 发票
Invoice(ctx context.Context, in *InvoiceReq) (res *InvoiceRes, err error)
2023-09-22 08:27:17 +00:00
// 支付成功
PaymentSuccessful(ctx context.Context, in *PaymentSuccessfulReq) (res *PaymentSuccessfulRes, err error)
2023-09-28 09:53:41 +00:00
// 关闭
Close(ctx context.Context, in *CloseReq) (res *CloseRes, err error)
2023-10-07 10:36:59 +00:00
// 删除
Delete(ctx context.Context, in *DeleteReq) (res *DeleteRes, err error)
2023-09-28 09:53:41 +00:00
// 支付超时订单自动关闭
CloseList(ctx context.Context, in *CloseListReq) (res *CloseListRes, err error)
2023-09-22 08:27:17 +00:00
}
PayInfo struct {
2023-09-28 02:05:55 +00:00
PayMethod string `json:"pay_method"` // 支付方式
PaymentMethod string `json:"payment_method"` // 交易方式
PayTime time.Time `json:"pay_time"` // 支付时间
Status gmodel.PayStatus `json:"status"` // 当前状态
StatusLink []gmodel.PayStatus `json:"status_link"` // 状态链路
TradeNo string `json:"trade_no"` // 支付交易号
2023-09-15 09:58:45 +00:00
}
2023-09-19 11:17:04 +00:00
OrderAddress struct {
Street string `json:"street"` // 详细地址
City string `json:"city"` // 城市
FirstName string `json:"first_name"` // 姓
LastName string `json:"last_name"` // 名
Mobile string `json:"mobile"` // 手机
State string `json:"state"` // 州
Suite string `json:"suite"` // 房号
ZipCode string `json:"zip_code"` // 邮编号码
2023-09-19 11:17:04 +00:00
}
2023-09-21 11:01:48 +00:00
OrderPay struct {
ClientSecret string `json:"client_secret"` // 支付方--秘钥
Country string `json:"country"` // 国家
Currency string `json:"currency"` // 货币
Metadata map[string]interface{} `json:"metadata"` // 额外参数
Method string `json:"method"` // 支付方--途径
OrderSn string `json:"order_sn"` // 订单编码
PayStage int64 `json:"pay_stage"` // 支付阶段
RedirectURL *string `json:"redirect_url"` // 支付方--重定向地址
Total OrderPayTotal `json:"total"` // 支付参数
}
// 支付参数
OrderPayTotal struct {
Amount int64 `json:"amount"` // 金额
Label string `json:"label"` // 标签
}
/* 发票 */
InvoiceReq struct {
UserId int64 `json:"user_id"`
OrderSn string `json:"order_sn"`
}
InvoiceRes struct {
ErrorCode basic.StatusResponse
InvoiceUrls []string `json:"invoice_urls"`
}
/* 发票 */
2023-10-07 10:36:59 +00:00
/* 删除订单 */
DeleteReq struct {
UserId int64 `json:"user_id"`
OrderSn string `json:"order_sn"`
}
DeleteRes struct {
2023-10-30 10:29:47 +00:00
ErrorCode basic.StatusResponse `json:"error_code"`
OrderSn string `json:"order_sn"`
2023-10-07 10:36:59 +00:00
}
/* 删除订单 */
2023-09-28 09:53:41 +00:00
/* 支付超时订单自动关闭 */
CloseListReq struct {
2023-10-30 10:29:47 +00:00
Type int64 `json:"type"` // type:1=关闭
2023-09-28 09:53:41 +00:00
}
CloseListRes struct{}
/* 支付超时订单自动关闭 */
/* 关闭 */
CloseReq struct {
2023-10-30 10:29:47 +00:00
Type int64 `json:"type"` // type:1=添加购物车
OrderSn string `json:"order_sn"`
UserId int64 `json:"user_id"`
2023-09-28 09:53:41 +00:00
}
CloseRes struct {
ErrorCode basic.StatusResponse
}
/* 关闭 */
2023-09-22 08:27:17 +00:00
/* 支付成功 */
PaymentSuccessfulReq struct {
2023-10-30 10:29:47 +00:00
EventId string `json:"event_id"`
PaymentMethod string `json:"payment_method"`
2023-09-25 10:28:53 +00:00
Charge *stripe.Charge
2023-09-22 08:27:17 +00:00
}
PaymentSuccessfulRes struct{}
/* 支付成功 */
2023-09-21 11:01:48 +00:00
/* 预支付--定金 */
CreatePrePaymentByDepositReq struct {
StripeKey string `json:"stripe_key"`
Currency string `json:"currency"`
Country string `json:"country"`
UserId int64 `json:"user_id"`
OrderSn string `json:"order_sn"`
DeliveryMethod int64 `json:"delivery_method"`
2023-09-22 02:39:02 +00:00
DeliveryAddress *OrderAddress `json:"delivery_address"`
2023-09-21 11:01:48 +00:00
}
CreatePrePaymentByDepositRes struct {
ErrorCode basic.StatusResponse
OrderDetail gmodel.OrderDetail
OrderPay OrderPay
}
/* 预支付--定金 */
2023-09-19 11:17:04 +00:00
2023-09-22 02:39:02 +00:00
/* 预支付--尾款 */
CreatePrePaymentByBalanceReq struct {
2023-09-22 03:00:54 +00:00
StripeKey string `json:"stripe_key"`
Currency string `json:"currency"`
Country string `json:"country"`
UserId int64 `json:"user_id"`
OrderSn string `json:"order_sn"`
2023-09-22 02:39:02 +00:00
}
CreatePrePaymentByBalanceRes struct {
ErrorCode basic.StatusResponse
OrderDetail gmodel.OrderDetail
OrderPay OrderPay
}
/* 预支付--尾款 */
2023-09-15 09:58:45 +00:00
/* 下单 */
CreateReq struct {
2023-09-20 07:07:12 +00:00
ExpectedDeliveryTime time.Time `json:"expected_delivery_time"` // 预计到货时间
2023-09-19 11:17:04 +00:00
ExchangeRate int64 `json:"exchange_rate"` // 换算汇率(厘)
CurrentCurrency string `json:"current_currency"` // 当前货币
OriginalCurrency string `json:"original_currency"` // 原始货币
UserId int64 `json:"user_id"`
CartIds []int64 `json:"cart_ids"`
DeliveryMethod int64 `json:"delivery_method"`
DeliveryAddress *OrderAddress `json:"delivery_address"` // 收货地址
2023-09-15 09:58:45 +00:00
}
CreateRes struct {
ErrorCode basic.StatusResponse
2023-10-30 10:29:47 +00:00
OrderSn string `json:"order_sn"`
2023-09-15 09:58:45 +00:00
}
/* 下单 */
2023-09-21 02:30:36 +00:00
/* 详情 */
DetailReq struct {
UserId int64 `json:"user_id"`
OrderSn string `json:"order_sn"`
}
DetailRes struct {
2023-09-21 11:01:48 +00:00
ErrorCode basic.StatusResponse
OrderDetail gmodel.OrderDetail
2023-09-27 09:23:58 +00:00
OrderDetailOriginal OrderDetailOriginal
}
OrderDetailOriginal struct {
Status *gmodel.OrderStatus
OrderAmount *gmodel.OrderAmount
OrderAddress *gmodel.OrderAddress
OrderProduct []gmodel.OrderProductInter
ShoppingCartSnapshot []gmodel.FsShoppingCart
ShoppingProductSnapshot []gmodel.RelaFsProduct
StatusLink []gmodel.OrderStatus
PayStatusLink []gmodel.PayStatus
2023-09-21 02:30:36 +00:00
}
/* 详情 */
2023-09-21 11:01:48 +00:00
/* 列表 */
ListReq struct {
UserId int64 `json:"user_id"`
DeliveryMethod int64 `json:"delivery_method"`
OrderCycle string `json:"order_cycle"`
CurrentPage int64 `json:"current_page"`
PerPage int64 `json:"per_page"`
}
ListRes struct {
OrderDetailList []gmodel.OrderDetail
Meta interface{}
}
/* 列表 */
2023-09-15 09:58:45 +00:00
)
// 订单发票
func (d *defaultOrder) Invoice(ctx context.Context, in *InvoiceReq) (res *InvoiceRes, err error) {
var errorCode basic.StatusResponse
var orderInfo gmodel.FsOrder
var receiptSnsResources []string
var receiptSnsDeposit string
var receiptSnsFinal string
var invoiceUrls []string
var orderTradeDeposit gmodel.FsOrderTrade
var orderTradeFinal gmodel.FsOrderTrade
model := d.MysqlConn.Where("is_del = ?", 0)
if in.UserId != 0 {
model = model.Where("user_id = ?", in.UserId)
}
if in.OrderSn != "" {
model = model.Where("order_sn = ?", in.OrderSn)
}
result := model.Take(&orderInfo)
if result.Error != nil {
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
logc.Errorf(ctx, "order invoice detail failed, err: %v", err)
return &InvoiceRes{
ErrorCode: errorCode,
}, result.Error
}
if *orderInfo.PayStatus == int64(constants.ORDER_PAY_STATUS_PAIDDEPOSIT) || *orderInfo.PayStatus == int64(constants.ORDER_PAY_STATUS_PAIDDREMAINING) {
// 查询支付账单
var orderTradeList []gmodel.FsOrderTrade
model1 := d.MysqlConn
if in.OrderSn != "" {
model1 = model1.Where("order_sn = ?", in.OrderSn)
}
result1 := model1.Find(&orderTradeList)
if result1.Error != nil {
if errors.Is(result1.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
logc.Errorf(ctx, "order invoice trade failed, err: %v", err)
return &InvoiceRes{
ErrorCode: errorCode,
}, result1.Error
}
if len(orderTradeList) > 0 {
for _, orderTrade := range orderTradeList {
receiptSnsResources = append(receiptSnsResources, hash.JsonHashKey(*orderTrade.ReceiptSn))
if *orderTrade.PayStage == 1 {
receiptSnsDeposit = *orderTrade.ReceiptSn
orderTradeDeposit = orderTrade
}
if *orderTrade.PayStage == 2 {
receiptSnsFinal = *orderTrade.ReceiptSn
orderTradeFinal = orderTrade
}
}
// 查询支付账单
var resourceList []gmodel.FsResource
result2 := d.MysqlConn.Where("resource_id in ?", receiptSnsResources).Find(&resourceList)
if result2.Error != nil {
errorCode = *basic.CodeServiceErr
logc.Errorf(ctx, "order invoice esource failed, err: %v", err)
return &InvoiceRes{
ErrorCode: errorCode,
}, result1.Error
}
resourceListLen := len(resourceList)
for _, resource := range resourceList {
invoiceUrls = append(invoiceUrls, *resource.ResourceUrl)
}
if *orderInfo.PayStatus == int64(constants.ORDER_PAY_STATUS_PAIDDEPOSIT) {
if resourceListLen == 1 {
return &InvoiceRes{
ErrorCode: errorCode,
InvoiceUrls: invoiceUrls,
}, nil
}
if resourceListLen == 0 {
receiptSnsFinal = ""
}
}
if *orderInfo.PayStatus == int64(constants.ORDER_PAY_STATUS_PAIDDREMAINING) {
if resourceListLen == 2 {
return &InvoiceRes{
ErrorCode: errorCode,
InvoiceUrls: invoiceUrls,
}, nil
}
if resourceListLen == 1 {
receiptSnsDeposit = ""
}
}
} else {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
return &InvoiceRes{
ErrorCode: errorCode,
}, errors.New("get order invoice failed, not found")
}
} else {
err = errors.New("get order invoice failed, pay status is illegality")
errorCode = *basic.CodeErrOrderInvoiceStatusIllegality
return &InvoiceRes{
ErrorCode: errorCode,
}, err
}
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &orderInfo, 1)
if err != nil {
logc.Errorf(ctx, "order invoice detail handler failed, err: %v", err)
errorCode = *basic.CodeServiceErr
return &InvoiceRes{
ErrorCode: errorCode,
}, err
} else {
var model001 = constants.INVOICE_TEMPLATE_01
var model002 string
var model003 string
var model004 string
var model005 string
var model006 = constants.INVOICE_TEMPLATE_06
ctimeDate := orderInfo.Ctime.Format("2006/01/02")
var name string
var city string
var street string
var state string
2023-10-18 04:10:29 +00:00
var suite string
var zipCode string
if ress.OrderDetail.DeliveryAddress != nil {
name = fmt.Sprintf("%s %s", ress.OrderDetail.DeliveryAddress.FirstName, ress.OrderDetail.DeliveryAddress.LastName)
street = ress.OrderDetail.DeliveryAddress.Street
city = ress.OrderDetail.DeliveryAddress.City
state = ress.OrderDetail.DeliveryAddress.State
2023-10-18 04:10:29 +00:00
suite = ress.OrderDetail.DeliveryAddress.Suite
zipCode = ress.OrderDetail.DeliveryAddress.ZipCode
2023-10-24 07:22:39 +00:00
} else {
// 邮箱
2023-10-24 07:37:47 +00:00
modelUser := d.MysqlConn.Model(&gmodel.FsUser{})
2023-10-24 07:22:39 +00:00
if in.UserId != 0 {
2023-10-24 07:37:47 +00:00
modelUser = modelUser.Where("id = ?", in.UserId)
2023-10-24 07:22:39 +00:00
}
2023-10-24 07:37:47 +00:00
var userInfo = &gmodel.FsUser{}
result1 := modelUser.Debug().Take(&userInfo)
2023-10-24 07:22:39 +00:00
if result1.Error != nil {
if errors.Is(result1.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
logc.Errorf(ctx, "order invoice trade failed, err: %v", err)
return &InvoiceRes{
ErrorCode: errorCode,
}, result1.Error
}
2023-10-24 07:37:47 +00:00
if userInfo.Email != nil {
name = *userInfo.Email
}
}
var products string
for _, orderProduct := range ress.OrderDetail.OrderProduct {
var model00301 = constants.INVOICE_TEMPLATE_0301
2023-10-24 07:22:39 +00:00
2023-10-30 09:41:53 +00:00
var priceStr = orderProduct.ItemPrice.Current.CurrentAmount.(string)
2023-10-24 07:22:39 +00:00
var price = fmt.Sprintf("%s%s", constants.OrderCurrencyMessage[constants.Currency(orderProduct.ItemPrice.Current.CurrentCurrency)], priceStr)
2023-10-30 09:41:53 +00:00
var priceTotalStr = orderProduct.TotalPrice.Current.CurrentAmount.(string)
2023-10-24 07:22:39 +00:00
var priceTotal = fmt.Sprintf("%s%s", constants.OrderCurrencyMessage[constants.Currency(orderProduct.TotalPrice.Current.CurrentCurrency)], priceTotalStr)
2023-10-30 09:41:53 +00:00
productNum := orderProduct.PurchaseQuantity.Current.(string)
var productNumStr = productNum
2023-10-24 07:22:39 +00:00
2023-11-09 06:52:31 +00:00
var productsInfo = fmt.Sprintf(model00301, orderProduct.ProductName, price, productNumStr, orderProduct.ProductWebsiteUnit.Complex, priceTotal)
products = products + productsInfo
}
2023-10-18 04:10:29 +00:00
model003 = fmt.Sprintf(constants.INVOICE_TEMPLATE_03, products)
2023-10-30 09:41:53 +00:00
var subtotalStr = ress.OrderDetail.OrderAmount.Subtotal.Current.CurrentAmount.(string)
2023-10-24 07:22:39 +00:00
var subtotal = fmt.Sprintf("%s%s", constants.OrderCurrencyMessage[constants.Currency(ress.OrderDetail.OrderAmount.Subtotal.Current.CurrentCurrency)], subtotalStr)
var taxStr = "0.00"
if ress.OrderDetail.OrderAmount.Tax.Current.CurrentAmount != nil {
taxStr = ress.OrderDetail.OrderAmount.Tax.Current.CurrentAmount.(string)
}
var taxCurrency string = constants.OrderCurrencyMessage[constants.Currency(ress.OrderDetail.OrderAmount.Tax.Current.CurrentCurrency)]
if taxCurrency == "" {
taxCurrency = constants.OrderCurrencyMessage[constants.Currency(ress.OrderDetail.OrderAmount.Total.Current.CurrentCurrency)]
}
var tax = fmt.Sprintf("%s%s", taxCurrency, taxStr)
2023-10-30 09:41:53 +00:00
var totalStr = ress.OrderDetail.OrderAmount.Total.Current.CurrentAmount.(string)
2023-10-24 07:22:39 +00:00
var total = fmt.Sprintf("%s%s", constants.OrderCurrencyMessage[constants.Currency(ress.OrderDetail.OrderAmount.Total.Current.CurrentCurrency)], totalStr)
// 生成收据发票--首款
if receiptSnsDeposit != "" {
2023-10-18 04:10:29 +00:00
model002 = fmt.Sprintf(constants.INVOICE_TEMPLATE_02, receiptSnsDeposit, name, ctimeDate, street+" "+suite, city, state+zipCode)
2023-10-30 09:41:53 +00:00
var payAmountStr01 = ress.OrderDetail.OrderAmount.Deposit.PayAmount.Current.CurrentAmount.(string)
v7 := "Deposit Requested"
2023-10-24 07:22:39 +00:00
v8 := fmt.Sprintf("%s%s", constants.OrderCurrencyMessage[constants.Currency(ress.OrderDetail.OrderAmount.Deposit.PayAmount.Current.CurrentCurrency)], payAmountStr01)
v9 := "Deposit Due"
v10 := v8
2023-10-18 04:10:29 +00:00
model004 = fmt.Sprintf(constants.INVOICE_TEMPLATE_04, subtotal, tax, total, v7, v8, v9, v10)
cardSn := "****" + *orderTradeDeposit.CardSn
2023-10-18 04:10:29 +00:00
model005 = fmt.Sprintf(constants.INVOICE_TEMPLATE_05, *orderTradeDeposit.CardBrand, cardSn)
var content = model001 + model002 + model003 + model004 + model005 + model006
2023-10-24 07:22:39 +00:00
base64, err := pdf.HtmlToPdfBase64(content, "html", "Letter")
if err != nil {
logc.Errorf(ctx, "order invoice HtmlToPdfBase64 failed, err: %v", err)
errorCode = *basic.CodeServiceErr
return &InvoiceRes{
ErrorCode: errorCode,
}, err
}
// 根据hash 查询数据资源
var resourceId string = hash.JsonHashKey(receiptSnsDeposit)
// 上传文件
var upload = file.Upload{
Ctx: ctx,
MysqlConn: d.MysqlConn,
AwsSession: d.AwsSession,
}
uploadRes, err := upload.UploadFileByBase64(&file.UploadBaseReq{
FileHash: resourceId,
FileData: "data:application/pdf;base64," + base64,
UploadBucket: 1,
ApiType: 2,
UserId: *orderInfo.UserId,
GuestId: 0,
Source: "order_invoice",
Refresh: 1,
})
if err != nil {
if err != nil {
logc.Errorf(ctx, "order invoice UploadFileByBase64 failed, err: %v", err)
errorCode = *basic.CodeServiceErr
return &InvoiceRes{
ErrorCode: errorCode,
}, err
}
}
invoiceUrls = append(invoiceUrls, uploadRes.ResourceUrl)
}
// 生成收据发票--尾款
if receiptSnsFinal != "" {
2023-10-18 04:10:29 +00:00
model002 = fmt.Sprintf(constants.INVOICE_TEMPLATE_02, receiptSnsDeposit, name, ctimeDate, street+" "+suite, city, state+zipCode)
2023-10-30 09:41:53 +00:00
var payAmountStr02 = ress.OrderDetail.OrderAmount.RemainingBalance.PayAmount.Current.CurrentAmount.(string)
v7 := "Balance Requested"
2023-10-24 07:22:39 +00:00
v8 := fmt.Sprintf("%s%s", constants.OrderCurrencyMessage[constants.Currency(ress.OrderDetail.OrderAmount.RemainingBalance.PayAmount.Current.CurrentCurrency)], payAmountStr02)
v9 := "Balance Due"
v10 := v8
2023-10-18 04:10:29 +00:00
model004 = fmt.Sprintf(constants.INVOICE_TEMPLATE_04, subtotal, tax, total, v7, v8, v9, v10)
cardSn := "****" + *orderTradeFinal.CardSn
2023-10-18 04:10:29 +00:00
model005 = fmt.Sprintf(constants.INVOICE_TEMPLATE_05, *orderTradeDeposit.CardBrand, cardSn)
var content = model001 + model002 + model003 + model004 + model005 + model006
2023-10-24 07:22:39 +00:00
base64, err := pdf.HtmlToPdfBase64(content, "html", "Letter")
if err != nil {
logc.Errorf(ctx, "order invoice HtmlToPdfBase64 failed, err: %v", err)
errorCode = *basic.CodeServiceErr
return &InvoiceRes{
ErrorCode: errorCode,
}, err
}
// 根据hash 查询数据资源
var resourceId string = hash.JsonHashKey(receiptSnsDeposit)
// 上传文件
var upload = file.Upload{
Ctx: ctx,
MysqlConn: d.MysqlConn,
AwsSession: d.AwsSession,
}
uploadRes, err := upload.UploadFileByBase64(&file.UploadBaseReq{
FileHash: resourceId,
FileData: "data:application/pdf;base64," + base64,
UploadBucket: 1,
ApiType: 2,
UserId: *orderInfo.UserId,
GuestId: 0,
Source: "order_invoice",
Refresh: 1,
})
if err != nil {
if err != nil {
logc.Errorf(ctx, "order invoice UploadFileByBase64 failed, err: %v", err)
errorCode = *basic.CodeServiceErr
return &InvoiceRes{
ErrorCode: errorCode,
}, err
}
}
invoiceUrls = append(invoiceUrls, uploadRes.ResourceUrl)
}
return &InvoiceRes{
ErrorCode: errorCode,
InvoiceUrls: invoiceUrls,
}, nil
}
}
2023-10-07 10:36:59 +00:00
// 订单删除
func (d *defaultOrder) Delete(ctx context.Context, in *DeleteReq) (res *DeleteRes, err error) {
var errorCode basic.StatusResponse
err = d.MysqlConn.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
var orderInfo gmodel.FsOrder
model := tx
if in.UserId != 0 {
model = model.Where("user_id = ?", in.UserId)
}
if in.OrderSn != "" {
model = model.Where("order_sn = ?", in.OrderSn)
}
result := model.Take(&orderInfo)
if result.Error != nil {
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
logc.Errorf(ctx, "order delete failed, err: %v", err)
return result.Error
}
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &orderInfo, 0)
2023-10-07 10:36:59 +00:00
if err != nil {
logc.Errorf(ctx, "order delete failed DetailOrderDetailHandler,OrderSn:%s, err: %v", in.OrderSn, err)
return err
}
if !(*orderInfo.Status == int64(constants.ORDER_STATUS_CLOSE) || (*orderInfo.DeliveryMethod == int64(constants.DELIVERYMETHODDIRECTMAIL) && *orderInfo.Status == int64(constants.ORDER_STATUS_DIRECTMAIL_ARRIVED))) {
errorCode = *basic.CodeErrOrderDeleteStatusIllegality
err = errors.New("order delete failed, status is illegality")
logc.Errorf(ctx, "order delete failed, err: %v", err)
return err
}
// 状态链路
var ntime = time.Now().UTC()
var statusCode = constants.ORDER_STATUS_DELETE
var statusLink = append(ress.OrderDetailOriginal.StatusLink, gmodel.OrderStatus{
Ctime: &ntime,
Utime: &ntime,
StatusCode: statusCode,
StatusTitle: constants.OrderStatusMessage[statusCode],
})
statusLinkByte, err := json.Marshal(statusLink)
if err != nil {
logc.Errorf(ctx, "order delete failed Marshal statusLinkByte,OrderSn:%s, err: %v", in.OrderSn, err)
return err
}
var table = gmodel.NewAllModels(tx).FsOrder.TableName()
var resUpdate *gorm.DB
var resUpdateSql string = fmt.Sprintf("UPDATE %s SET `utime` = '%s',`is_del` = 1", table, ntime)
resUpdate = tx.Exec(fmt.Sprintf("%s ,`status_link`= JSON_MERGE_PATCH(`status_link`,?) WHERE `id` = %d", resUpdateSql, orderInfo.Id), statusLinkByte)
err = resUpdate.Error
if err != nil {
logc.Errorf(ctx, "order delete failed Update FsOrder,OrderSn:%s, err: %v", in.OrderSn, err)
return err
}
return nil
})
if err != nil {
logc.Errorf(ctx, "order delete failed, err: %v", err)
if errorCode.Code == 0 {
errorCode.Code = basic.CodeApiErr.Code
errorCode.Message = basic.CodeApiErr.Message
}
return &DeleteRes{
ErrorCode: errorCode,
}, err
}
return &DeleteRes{
ErrorCode: errorCode,
OrderSn: in.OrderSn,
}, err
}
2023-09-28 10:27:43 +00:00
// 订单关闭-支付超时
2023-09-28 09:53:41 +00:00
func (d *defaultOrder) CloseList(ctx context.Context, in *CloseListReq) (res *CloseListRes, err error) {
var orderList []gmodel.FsOrder
result := d.MysqlConn.Model(&gmodel.FsOrder{}).
Where("is_del = ?", 0).
Where("status = ?", int64(constants.ORDER_STATUS_UNPAIDDEPOSIT)).
Where("pay_status = ?", int64(constants.ORDER_PAY_STATUS_UNPAIDDEPOSIT)).
Find(&orderList)
if result.Error != nil {
2023-09-28 10:17:43 +00:00
logc.Errorf(ctx, "order CloseList failed, err: %v", err)
2023-09-28 09:53:41 +00:00
return nil, result.Error
}
2023-10-07 08:12:29 +00:00
for _, orderInfo := range orderList {
var ntime = time.Now().UTC()
2023-10-07 09:31:24 +00:00
var cptime = orderInfo.Ctime.Add(time.Minute * 30)
2023-10-07 08:12:29 +00:00
var dd = ntime.Unix() - cptime.Unix()
if in.Type == 1 {
2023-10-07 09:31:24 +00:00
fmt.Println("未支付超时时间: dd--", dd)
if dd >= 0 {
orderSn := *orderInfo.OrderSn
fmt.Println("即时任务: OrderSn--", orderSn)
2023-10-07 08:12:29 +00:00
ctx := context.Background()
logc.Infof(ctx, "order CloseList, orderSn: %s", orderSn)
d.Close(ctx, &CloseReq{
2023-10-07 09:31:24 +00:00
OrderSn: orderSn,
2023-10-07 08:12:29 +00:00
Type: 1,
})
2023-10-07 09:31:24 +00:00
} else {
ddd := math.Abs(float64(dd))
// 延时任务
time.AfterFunc(time.Second*time.Duration(ddd), func() {
orderSn := *orderInfo.OrderSn
fmt.Println("延时任务: OrderSn--", orderSn)
ctx := context.Background()
logc.Infof(ctx, "order CloseList, orderSn: %s", orderSn)
d.Close(ctx, &CloseReq{
OrderSn: orderSn,
Type: 1,
})
})
}
2023-10-07 08:12:29 +00:00
}
}
2023-09-28 09:53:41 +00:00
return nil, nil
}
// 关闭
func (d *defaultOrder) Close(ctx context.Context, in *CloseReq) (res *CloseRes, err error) {
var errorCode basic.StatusResponse
err = d.MysqlConn.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
var orderInfo gmodel.FsOrder
model := tx.Where("status = ?", int64(constants.ORDER_STATUS_UNPAIDDEPOSIT)).Where("pay_status = ?", int(constants.ORDER_PAY_STATUS_UNPAIDDEPOSIT))
2023-10-08 10:38:39 +00:00
if in.UserId != 0 {
model = model.Where("user_id = ?", in.UserId)
}
2023-09-28 09:53:41 +00:00
if in.OrderSn != "" {
2023-10-07 10:36:59 +00:00
model = model.Where("order_sn = ?", in.OrderSn)
2023-09-28 09:53:41 +00:00
}
result := model.Take(&orderInfo)
if result.Error != nil {
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
logc.Errorf(ctx, "order close failed, err: %v", err)
return result.Error
}
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &orderInfo, 0)
2023-09-28 09:53:41 +00:00
if err != nil {
logc.Errorf(ctx, "order close failed DetailOrderDetailHandler,OrderSn:%s, err: %v", in.OrderSn, err)
return err
}
// 更新状态、状态链路
var ntime = time.Now().UTC()
var statusCode = constants.ORDER_STATUS_CLOSE
var statusLink = order.UpdateOrderStatusLink(ress.OrderDetailOriginal.StatusLink, gmodel.OrderStatus{
Ctime: &ntime,
Utime: &ntime,
StatusCode: statusCode,
StatusTitle: constants.OrderStatusMessage[statusCode],
})
statusLinkByte, err := json.Marshal(statusLink)
if err != nil {
logc.Errorf(ctx, "order close failed Marshal statusLinkByte,OrderSn:%s, err: %v", in.OrderSn, err)
return err
}
var table = gmodel.NewAllModels(tx).FsOrder.TableName()
var resUpdate *gorm.DB
var resUpdateSql string = fmt.Sprintf("UPDATE %s SET `status` = %d , `utime` = '%s'", table, statusCode, ntime)
resUpdate = tx.Exec(fmt.Sprintf("%s ,`status_link`= JSON_MERGE_PATCH(`status_link`,?) WHERE `id` = %d", resUpdateSql, orderInfo.Id), statusLinkByte)
err = resUpdate.Error
if err != nil {
logc.Errorf(ctx, "order close failed Update FsOrder,OrderSn:%s, err: %v", in.OrderSn, err)
return err
}
// 新增购物车
if in.Type == 1 {
var users []gmodel.FsShoppingCart
for _, shoppingCart := range ress.OrderDetailOriginal.ShoppingCartSnapshot {
users = append(users, gmodel.FsShoppingCart{
UserId: shoppingCart.UserId,
ProductId: shoppingCart.ProductId,
TemplateId: shoppingCart.TemplateId,
ModelId: shoppingCart.ModelId,
SizeId: shoppingCart.SizeId,
LightId: shoppingCart.FittingId,
FittingId: shoppingCart.FittingId,
PurchaseQuantity: shoppingCart.PurchaseQuantity,
Snapshot: shoppingCart.Snapshot,
IsSelected: shoppingCart.IsSelected,
IsHighlyCustomized: shoppingCart.IsHighlyCustomized,
Ctime: &ntime,
Utime: &ntime,
})
}
resCreate := tx.Create(&users)
err = resCreate.Error
if err != nil {
logc.Errorf(ctx, "order close failed Create FsShoppingCart,OrderSn:%s, err: %v", in.OrderSn, err)
return err
}
}
return nil
})
if err != nil {
logc.Errorf(ctx, "order close failed, err: %v", err)
if errorCode.Code == 0 {
errorCode.Code = basic.CodeApiErr.Code
errorCode.Message = basic.CodeApiErr.Message
}
return &CloseRes{
ErrorCode: errorCode,
}, err
}
return &CloseRes{
ErrorCode: errorCode,
}, err
}
2023-09-22 08:27:17 +00:00
// 支付成功
func (d *defaultOrder) PaymentSuccessful(ctx context.Context, in *PaymentSuccessfulReq) (res *PaymentSuccessfulRes, err error) {
var orderSn string
var payStage string
var ok bool
var card string
var brand string
var country string
var currency string
var tradeSn string
var payAmount int64
var payTitle string
var payTime time.Time
2023-09-28 02:05:55 +00:00
var paymentMethod string
2023-09-26 02:56:57 +00:00
2023-09-28 02:05:55 +00:00
var payMethod int64 = 1
2023-09-26 02:56:57 +00:00
if in.PaymentMethod == "stripe" {
2023-09-28 02:05:55 +00:00
payMethod = 1
2023-09-26 02:56:57 +00:00
}
if in.PaymentMethod == "paypal" {
2023-09-28 02:05:55 +00:00
payMethod = 2
2023-09-26 02:56:57 +00:00
}
2023-09-25 10:28:53 +00:00
if in.Charge != nil {
charge := in.Charge
orderSn, ok = charge.Metadata["order_sn"]
2023-09-22 08:27:17 +00:00
if !ok || orderSn == "" {
err = errors.New("order_sn is empty")
logc.Errorf(ctx, "PaymentSuccessful failed param, eventId:%s,err:%v", in.EventId, err)
return &PaymentSuccessfulRes{}, err
}
2023-09-25 10:28:53 +00:00
payStage, ok = charge.Metadata["pay_stage"]
2023-09-22 08:27:17 +00:00
if !ok || payStage == "" {
err = errors.New("pay_stage is empty")
logc.Errorf(ctx, "PaymentSuccessful failed param, eventId:%s,err:%v", in.EventId, err)
return &PaymentSuccessfulRes{}, err
}
2023-09-25 10:28:53 +00:00
country, ok = charge.Metadata["country"]
2023-09-22 08:27:17 +00:00
if !ok || country == "" {
err = errors.New("country is empty")
logc.Errorf(ctx, "PaymentSuccessful failed param, eventId:%s,err:%v", in.EventId, err)
return &PaymentSuccessfulRes{}, err
}
2023-09-25 10:28:53 +00:00
if charge.PaymentMethodDetails != nil {
if charge.PaymentMethodDetails.Card != nil {
if charge.PaymentMethodDetails.Card.Last4 != "" {
card = charge.PaymentMethodDetails.Card.Last4
2023-09-22 08:27:17 +00:00
}
2023-09-25 10:28:53 +00:00
if charge.PaymentMethodDetails.Card.Brand != "" {
brand = string(charge.PaymentMethodDetails.Card.Brand)
2023-09-22 08:27:17 +00:00
}
}
2023-09-28 02:05:55 +00:00
if charge.PaymentMethodDetails.Type != "" {
paymentMethod = string(charge.PaymentMethodDetails.Type)
}
2023-09-22 08:27:17 +00:00
}
2023-09-25 10:28:53 +00:00
if charge.Currency != "" {
currency = string(charge.Currency)
2023-09-22 08:27:17 +00:00
}
2023-09-25 10:28:53 +00:00
tradeSn = charge.ID
payAmount = charge.Amount
payTitle = charge.Description
2023-09-26 09:32:21 +00:00
payTime = time.Unix(charge.Created, 0).UTC()
2023-09-22 08:27:17 +00:00
}
err = d.MysqlConn.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
var orderInfo gmodel.FsOrder
result := tx.Where("is_del = ?", 0).Where("order_sn = ?", orderSn).Take(&orderInfo)
err = result.Error
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "PaymentSuccessful failed order Take, eventId:%s,err: %v", in.EventId, err)
2023-09-22 08:27:17 +00:00
return err
}
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &orderInfo, 0)
2023-09-22 08:27:17 +00:00
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "PaymentSuccessful failed DetailOrderDetailHandler,eventId:%s, err: %v", in.EventId, err)
2023-09-22 08:27:17 +00:00
return err
}
var ntime = time.Now().UTC()
2023-09-27 09:23:58 +00:00
if (payStage == "deposit" && *orderInfo.PayStatus == int64(constants.ORDER_PAY_STATUS_UNPAIDDEPOSIT)) || (payStage == "remaining_balance" && *orderInfo.PayStatus == int64(constants.ORDER_PAY_STATUS_PAIDDEPOSIT)) {
var payStatus = int64(constants.PAY_STATUS_PAID)
2023-09-22 08:27:17 +00:00
var payStageInt int64
var orderPayStatusCode constants.OrderPayStatusCode
// 订单状态--当前
var status gmodel.OrderStatus
var statusLink []gmodel.OrderStatus
2023-09-27 09:23:58 +00:00
var payStatusLink []gmodel.PayStatus
var orderAmount = make(map[string]interface{})
2023-09-22 08:27:17 +00:00
var payInfo PayInfo
2023-09-28 02:05:55 +00:00
payInfo.PayMethod = paymentMethod
payInfo.PaymentMethod = in.PaymentMethod
2023-09-22 08:27:17 +00:00
payInfo.PayTime = payTime
payInfo.TradeNo = tradeSn
2023-09-26 02:56:57 +00:00
// 当前状态
var statusCode constants.OrderStatusCode
var statusCodePre constants.OrderStatusCode
2023-09-22 08:27:17 +00:00
if payStage == "deposit" {
2023-09-26 02:56:57 +00:00
if *orderInfo.DeliveryMethod == constants.DELIVERYMETHODDIRECTMAIL {
// 直邮
2023-09-26 10:46:44 +00:00
statusCode = constants.ORDER_STATUS_DIRECTMAIL_ORDERED
2023-09-26 02:56:57 +00:00
}
if *orderInfo.DeliveryMethod == constants.DELIVERYMETHODDSCLOUDSTORE {
// 云仓
2023-09-26 10:46:44 +00:00
statusCode = constants.ORDER_STATUS_CLOUDSTORE_ORDERED
2023-09-26 02:56:57 +00:00
}
2023-09-22 08:27:17 +00:00
payStageInt = 1
2023-09-27 09:23:58 +00:00
orderPayStatusCode = constants.ORDER_PAY_STATUS_PAIDDEPOSIT
2023-09-22 08:27:17 +00:00
status = gmodel.OrderStatus{
Ctime: &ntime,
Utime: &ntime,
2023-09-26 02:56:57 +00:00
StatusCode: statusCode,
StatusTitle: constants.OrderStatusMessage[statusCode],
2023-09-22 08:27:17 +00:00
}
2023-09-27 09:23:58 +00:00
statusLink = order.UpdateOrderStatusLink(ress.OrderDetailOriginal.StatusLink, gmodel.OrderStatus{
2023-09-26 07:25:05 +00:00
Ctime: &ntime,
2023-09-22 08:27:17 +00:00
Utime: &ntime,
2023-09-26 02:56:57 +00:00
StatusCode: statusCode,
StatusTitle: constants.OrderStatusMessage[statusCode],
2023-09-22 08:27:17 +00:00
})
payInfo.Status = gmodel.PayStatus{
2023-09-27 09:23:58 +00:00
StatusCode: int64(constants.PAY_STATUS_PAID),
StatusTitle: constants.PayStatusMessage[constants.PAY_STATUS_PAID],
2023-09-22 08:27:17 +00:00
}
2023-09-27 09:23:58 +00:00
payInfo.StatusLink = append(ress.OrderDetailOriginal.OrderAmount.Deposit.StatusLink, payInfo.Status)
orderAmount["deposit"] = payInfo
2023-09-22 08:27:17 +00:00
}
if payStage == "remaining_balance" {
2023-09-26 02:56:57 +00:00
if *orderInfo.DeliveryMethod == constants.DELIVERYMETHODDIRECTMAIL {
// 直邮
2023-09-26 10:46:44 +00:00
statusCodePre = constants.ORDER_STATUS_DIRECTMAIL_ORDERED
statusCode = constants.ORDER_STATUS_DIRECTMAIL_ORDEREDMAINING
2023-09-26 02:56:57 +00:00
}
if *orderInfo.DeliveryMethod == constants.DELIVERYMETHODDSCLOUDSTORE {
// 云仓
2023-09-26 10:46:44 +00:00
statusCodePre = constants.ORDER_STATUS_CLOUDSTORE_ORDERED
statusCode = constants.ORDER_STATUS_CLOUDSTORE_ORDEREDMAINING
2023-09-26 02:56:57 +00:00
}
2023-09-22 08:27:17 +00:00
payStageInt = 2
2023-09-27 09:23:58 +00:00
orderPayStatusCode = constants.ORDER_PAY_STATUS_PAIDDREMAINING
2023-09-26 02:56:57 +00:00
var statusChildren []*gmodel.OrderStatus
// 更新订单状态链路--子状态
2023-09-27 09:23:58 +00:00
for oStatusLinkKey, oStatusLink := range ress.OrderDetailOriginal.StatusLink {
2023-09-26 02:56:57 +00:00
if oStatusLink.StatusCode == statusCodePre {
statusChildren = append(oStatusLink.Children, &gmodel.OrderStatus{
Ctime: &ntime,
Utime: &ntime,
StatusCode: statusCode,
StatusTitle: constants.OrderStatusMessage[statusCode],
})
2023-09-27 09:23:58 +00:00
ress.OrderDetailOriginal.StatusLink[oStatusLinkKey].Children = statusChildren
2023-09-26 02:56:57 +00:00
}
2023-09-22 08:27:17 +00:00
}
2023-09-27 09:23:58 +00:00
statusLink = ress.OrderDetailOriginal.StatusLink
2023-09-26 02:56:57 +00:00
2023-09-27 09:23:58 +00:00
if ress.OrderDetailOriginal.Status.StatusCode == constants.ORDER_STATUS_DIRECTMAIL_ORDERED || ress.OrderDetailOriginal.Status.StatusCode == constants.ORDER_STATUS_CLOUDSTORE_ORDERED {
status = *ress.OrderDetailOriginal.Status
2023-09-26 02:56:57 +00:00
status.Children = statusChildren
}
2023-09-26 09:32:21 +00:00
payInfo.Status = gmodel.PayStatus{
2023-09-27 09:23:58 +00:00
StatusCode: int64(constants.PAY_STATUS_PAID),
StatusTitle: constants.PayStatusMessage[constants.PAY_STATUS_PAID],
2023-09-26 09:32:21 +00:00
}
2023-09-26 02:56:57 +00:00
2023-09-27 09:23:58 +00:00
payInfo.StatusLink = append(ress.OrderDetailOriginal.OrderAmount.RemainingBalance.StatusLink, payInfo.Status)
orderAmount["remaining_balance"] = payInfo
}
2023-09-27 09:51:25 +00:00
payStatusLink = append(ress.OrderDetailOriginal.PayStatusLink, gmodel.PayStatus{
StatusCode: int64(orderPayStatusCode),
StatusTitle: constants.OrderPayStatusMessage[orderPayStatusCode],
})
2023-09-27 09:40:44 +00:00
2023-09-27 09:23:58 +00:00
orderAmountByte, err := json.Marshal(orderAmount)
if err != nil {
logc.Errorf(ctx, "PaymentSuccessful failed Marshal orderAmount,eventId:%s, err: %v", in.EventId, err)
return err
}
statusLinkByte, err := json.Marshal(statusLink)
if err != nil {
logc.Errorf(ctx, "PaymentSuccessful failed Marshal statusLink,eventId:%s, err: %v", in.EventId, err)
return err
}
payStatusLinkByte, err := json.Marshal(payStatusLink)
if err != nil {
logc.Errorf(ctx, "PaymentSuccessful failed Marshal payStatusLink,eventId:%s, err: %v", in.EventId, err)
return err
2023-09-22 08:27:17 +00:00
}
// 新增交易信息
receiptSn := order.GenerateReceiptNumber()
2023-09-22 08:27:17 +00:00
tx.Create(&gmodel.FsOrderTrade{
UserId: orderInfo.UserId,
OrderSn: &orderSn,
OrderSource: orderInfo.OrderSource,
TradeSn: &tradeSn,
PayAmount: &payAmount,
PayStatus: &payStatus,
2023-09-28 02:05:55 +00:00
PaymentMethod: &payMethod,
2023-09-22 08:27:17 +00:00
PayStage: &payStageInt,
CardSn: &card,
CardBrand: &brand,
Country: &country,
Currency: &currency,
Ctime: &ntime,
Utime: &ntime,
PayTitle: &payTitle,
ReceiptSn: &receiptSn,
2023-09-22 08:27:17 +00:00
})
2023-09-27 09:23:58 +00:00
// 更新数据库
2023-09-28 09:53:41 +00:00
var table = gmodel.NewAllModels(tx).FsOrder.TableName()
2023-09-27 09:23:58 +00:00
var resUpdate *gorm.DB
var resUpdateSql string
2023-09-26 10:46:44 +00:00
if *orderInfo.Status == int64(constants.ORDER_STATUS_UNPAIDDEPOSIT) {
2023-09-27 09:23:58 +00:00
resUpdateSql = fmt.Sprintf("UPDATE %s SET `status` = %d,`pay_status` = %d , `utime` = '%s'", table, statusCode, orderPayStatusCode, ntime)
2023-09-26 07:29:33 +00:00
} else {
2023-09-27 09:23:58 +00:00
resUpdateSql = fmt.Sprintf("UPDATE %s SET `pay_status` = %d , `utime` = '%s'", table, orderPayStatusCode, ntime)
2023-09-26 07:29:33 +00:00
}
2023-09-28 09:53:41 +00:00
resUpdate = tx.Exec(fmt.Sprintf("%s ,`status_link`= JSON_MERGE_PATCH(`status_link`,?),`pay_status_link`= JSON_MERGE_PATCH(`pay_status_link`,?),`order_amount`= JSON_MERGE_PATCH(`order_amount`,?) WHERE `id` = %d", resUpdateSql, orderInfo.Id), statusLinkByte, payStatusLinkByte, orderAmountByte)
2023-09-26 07:29:33 +00:00
2023-09-27 09:23:58 +00:00
err = resUpdate.Error
if err != nil {
logc.Errorf(ctx, "PaymentSuccessful failed resUpdate,eventId:%s, err: %v", in.EventId, err)
return err
2023-09-22 08:27:17 +00:00
}
}
return nil
})
return &PaymentSuccessfulRes{}, nil
}
2023-09-22 02:39:02 +00:00
// 预支付--尾款
func (d *defaultOrder) CreatePrePaymentByBalance(ctx context.Context, in *CreatePrePaymentByBalanceReq) (res *CreatePrePaymentByBalanceRes, err error) {
2023-09-22 03:00:54 +00:00
var errorCode basic.StatusResponse
2023-09-27 09:23:58 +00:00
var orderInfo gmodel.FsOrder
2023-09-22 03:00:54 +00:00
model := d.MysqlConn.Where("is_del = ?", 0)
if in.UserId != 0 {
2023-10-07 10:36:59 +00:00
model = model.Where("user_id = ?", in.UserId)
2023-09-22 03:00:54 +00:00
}
if in.OrderSn != "" {
2023-10-07 10:36:59 +00:00
model = model.Where("order_sn = ?", in.OrderSn)
2023-09-22 03:00:54 +00:00
}
2023-09-27 09:23:58 +00:00
result := model.Take(&orderInfo)
2023-09-22 03:00:54 +00:00
if result.Error != nil {
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment balance failed, err: %v", err)
2023-09-22 03:00:54 +00:00
return &CreatePrePaymentByBalanceRes{
ErrorCode: errorCode,
}, result.Error
}
// 非未支付
2023-09-27 09:23:58 +00:00
if *orderInfo.PayStatus != int64(constants.ORDER_PAY_STATUS_PAIDDEPOSIT) {
2023-09-22 03:00:54 +00:00
errorCode = *basic.CodeErrOrderCreatePrePaymentNoUnPaid
err = errors.New("order balance pay status is not unPaid")
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment balance failed, err: %v", err)
2023-09-22 03:00:54 +00:00
return &CreatePrePaymentByBalanceRes{
ErrorCode: errorCode,
}, err
}
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &orderInfo, 1)
2023-09-22 03:00:54 +00:00
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment balance failed DetailOrderDetailHandler, err: %v", err)
2023-09-22 03:00:54 +00:00
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByBalanceRes{
ErrorCode: errorCode,
}, err
}
// 支付初始化
amount := int64(ress.OrderDetailOriginal.OrderAmount.RemainingBalance.PayAmount.Current.CurrentAmount.(float64) / float64(10))
payConfig := &pay.Config{}
payConfig.Stripe.PayType = "intent"
payConfig.Stripe.Key = in.StripeKey
2023-11-02 07:20:32 +00:00
var metadata = make(map[string]string, 4)
2023-09-22 08:27:17 +00:00
metadata["model"] = "product_order"
metadata["order_sn"] = in.OrderSn
metadata["pay_stage"] = "remaining_balance"
metadata["country"] = in.Country
2023-09-22 03:00:54 +00:00
var generatePrepaymentReq = &pay.GeneratePrepaymentReq{
2023-09-22 08:27:17 +00:00
Metadata: metadata,
2023-09-22 03:00:54 +00:00
ProductName: "支付尾款后期统一调整",
Amount: amount,
Currency: "usd",
Quantity: 1,
ProductDescription: "支付尾款后期统一调整",
}
payDriver := pay.NewPayDriver(1, payConfig)
prepaymentRes, err := payDriver.GeneratePrepayment(generatePrepaymentReq)
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment balance failed GeneratePrepayment, err: %v", err)
2023-09-22 03:00:54 +00:00
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByBalanceRes{
ErrorCode: errorCode,
}, nil
}
return &CreatePrePaymentByBalanceRes{
OrderDetail: ress.OrderDetail,
OrderPay: OrderPay{
ClientSecret: prepaymentRes.ClientSecret,
Country: in.Country,
Currency: in.Currency,
Method: payConfig.Stripe.PayType,
OrderSn: in.OrderSn,
2023-09-26 09:18:04 +00:00
PayStage: 2,
2023-09-22 03:00:54 +00:00
Total: OrderPayTotal{
Amount: amount,
Label: "支付尾款后期统一调整",
},
},
}, nil
2023-09-22 02:39:02 +00:00
}
2023-09-21 11:01:48 +00:00
// 预支付--定金
func (d *defaultOrder) CreatePrePaymentByDeposit(ctx context.Context, in *CreatePrePaymentByDepositReq) (res *CreatePrePaymentByDepositRes, err error) {
var errorCode basic.StatusResponse
2023-09-27 09:23:58 +00:00
var orderInfo gmodel.FsOrder
2023-09-21 11:01:48 +00:00
model := d.MysqlConn.Where("is_del = ?", 0)
if in.UserId != 0 {
2023-10-07 10:36:59 +00:00
model = model.Where("user_id = ?", in.UserId)
2023-09-21 11:01:48 +00:00
}
if in.OrderSn != "" {
2023-10-07 10:36:59 +00:00
model = model.Where("order_sn = ?", in.OrderSn)
2023-09-21 11:01:48 +00:00
}
2023-09-27 09:23:58 +00:00
result := model.Take(&orderInfo)
2023-09-21 02:30:36 +00:00
if result.Error != nil {
2023-09-21 11:01:48 +00:00
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment deposit failed, err: %v", err)
2023-09-21 11:01:48 +00:00
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, result.Error
2023-09-21 02:30:36 +00:00
}
2023-09-21 11:01:48 +00:00
// 非未支付
2023-09-27 09:23:58 +00:00
if *orderInfo.PayStatus != int64(constants.ORDER_PAY_STATUS_UNPAIDDEPOSIT) {
2023-09-22 03:00:54 +00:00
errorCode = *basic.CodeErrOrderCreatePrePaymentNoUnPaid
2023-09-21 11:01:48 +00:00
err = errors.New("order pay status is not unPaidDeposit")
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment deposit failed, err: %v", err)
2023-09-21 11:01:48 +00:00
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, err
}
// 是否超时支付
ntime := time.Now().UTC()
2023-09-27 09:23:58 +00:00
ctime := *orderInfo.Ctime
2023-09-21 11:01:48 +00:00
ctimeTimeOut := ctime.Add(30 * time.Minute).UTC().Unix()
ntimeTimeOut := ntime.Unix()
2023-09-22 03:00:54 +00:00
// 测试超时支付不限制
2023-10-07 08:13:08 +00:00
if ctimeTimeOut < ntimeTimeOut {
// if ctimeTimeOut == ntimeTimeOut {
2023-09-21 11:01:48 +00:00
errorCode = *basic.CodeErrOrderCreatePrePaymentTimeout
err = errors.New("order pay timeout")
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment deposit failed, err: %v", err)
2023-09-21 11:01:48 +00:00
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, err
}
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &orderInfo, 1)
2023-09-21 06:26:14 +00:00
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment deposit failed DetailOrderDetailHandler, err: %v", err)
2023-09-21 11:01:48 +00:00
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, err
2023-09-21 06:26:14 +00:00
}
2023-09-21 11:01:48 +00:00
var orderAddress *gmodel.OrderAddress
2023-09-27 09:23:58 +00:00
var orderAddressByte []byte
var statusLinkByte []byte
2023-10-16 09:42:03 +00:00
if in.DeliveryMethod == constants.DELIVERYMETHODDIRECTMAIL && in.DeliveryAddress != nil {
2023-09-22 04:12:33 +00:00
orderAddress = &gmodel.OrderAddress{
Street: in.DeliveryAddress.Street,
City: in.DeliveryAddress.City,
FirstName: in.DeliveryAddress.FirstName,
LastName: in.DeliveryAddress.LastName,
Mobile: in.DeliveryAddress.Mobile,
State: in.DeliveryAddress.State,
Suite: in.DeliveryAddress.Suite,
ZipCode: in.DeliveryAddress.ZipCode,
2023-09-21 11:01:48 +00:00
}
2023-09-27 09:23:58 +00:00
orderAddressByte, err = json.Marshal(orderAddress)
2023-09-22 02:39:02 +00:00
if err != nil {
2023-09-27 09:23:58 +00:00
logc.Errorf(ctx, "create prePayment deposit failed orderAddressByte, err: %v", err)
2023-09-22 02:39:02 +00:00
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
2023-09-27 09:23:58 +00:00
}, err
2023-09-22 02:39:02 +00:00
}
2023-10-16 09:42:03 +00:00
} else {
var orderAddress *gmodel.OrderAddress
orderAddressByte, err = json.Marshal(orderAddress)
if err != nil {
logc.Errorf(ctx, "create prePayment deposit failed orderAddressByte, err: %v", err)
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, err
}
2023-09-22 02:39:02 +00:00
}
2023-09-27 09:23:58 +00:00
// 订单状态--链路
tPlus60Days := ntime.AddDate(0, 0, 60).UTC()
var statusLink = order.GenerateOrderStatusLink(in.DeliveryMethod, ntime, tPlus60Days)
statusLinkByte, err = json.Marshal(statusLink)
if err != nil {
logc.Errorf(ctx, "create prePayment deposit failed statusLinkByte, err: %v", err)
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, err
}
// 更新数据库
var table = gmodel.NewAllModels(d.MysqlConn).FsOrder.TableName()
2023-10-16 09:42:03 +00:00
var resUpdate = d.MysqlConn.Exec(fmt.Sprintf("UPDATE %s SET `delivery_method` = ? , `utime` = ?, `status_link` = JSON_MERGE_PATCH(`status_link`,?), `order_address` = JSON_MERGE_PATCH(`order_address`,?) WHERE `id` = ?", table), in.DeliveryMethod, ntime, statusLinkByte, orderAddressByte, orderInfo.Id)
2023-09-27 09:23:58 +00:00
err = resUpdate.Error
if err != nil {
logc.Errorf(ctx, "create prePayment deposit failed update Exec, err: %v", err)
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, err
}
2023-09-22 02:39:02 +00:00
ress.OrderDetail.OrderInfo.Utime = &ntime
ress.OrderDetail.OrderInfo.DeliveryMethod = in.DeliveryMethod
2023-09-27 09:23:58 +00:00
ress.OrderDetail.OrderInfo.StatusLink = statusLink
2023-09-22 02:39:02 +00:00
ress.OrderDetail.DeliveryAddress = orderAddress
2023-09-21 11:01:48 +00:00
// 支付初始化
amount := int64(ress.OrderDetailOriginal.OrderAmount.Deposit.PayAmount.Current.CurrentAmount.(float64) / float64(10))
payConfig := &pay.Config{}
payConfig.Stripe.PayType = "intent"
payConfig.Stripe.Key = in.StripeKey
2023-11-02 07:20:32 +00:00
var metadata = make(map[string]string, 4)
2023-09-22 08:27:17 +00:00
metadata["model"] = "product_order"
metadata["order_sn"] = in.OrderSn
metadata["pay_stage"] = "deposit"
metadata["country"] = in.Country
2023-09-21 11:01:48 +00:00
var generatePrepaymentReq = &pay.GeneratePrepaymentReq{
2023-09-22 08:27:17 +00:00
Metadata: metadata,
2023-09-22 03:00:54 +00:00
ProductName: "支付首款",
2023-09-21 11:01:48 +00:00
Amount: amount,
Currency: "usd",
Quantity: 1,
2023-09-22 03:00:54 +00:00
ProductDescription: "支付首款",
2023-09-21 11:01:48 +00:00
}
payDriver := pay.NewPayDriver(1, payConfig)
prepaymentRes, err := payDriver.GeneratePrepayment(generatePrepaymentReq)
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "create prePayment deposit failed GeneratePrepayment, err: %v", err)
2023-09-21 11:01:48 +00:00
errorCode = *basic.CodeServiceErr
return &CreatePrePaymentByDepositRes{
ErrorCode: errorCode,
}, nil
}
return &CreatePrePaymentByDepositRes{
OrderDetail: ress.OrderDetail,
OrderPay: OrderPay{
ClientSecret: prepaymentRes.ClientSecret,
Country: in.Country,
Currency: in.Currency,
Method: payConfig.Stripe.PayType,
OrderSn: in.OrderSn,
PayStage: 1,
Total: OrderPayTotal{
Amount: amount,
2023-11-02 07:22:56 +00:00
Label: "支付首款后期统一调整",
2023-09-21 11:01:48 +00:00
},
},
2023-09-21 06:26:14 +00:00
}, nil
2023-09-21 02:30:36 +00:00
}
2023-09-21 11:01:48 +00:00
// 列表
func (d *defaultOrder) List(ctx context.Context, in *ListReq) (res *ListRes, err error) {
var orderList []gmodel.FsOrder
model := d.MysqlConn.Model(&gmodel.FsOrder{}).Where("is_del = ?", 0)
2023-10-08 02:23:39 +00:00
model = model.Where("pay_status > ?", 0)
2023-09-21 11:01:48 +00:00
if in.UserId != 0 {
2023-10-07 10:36:59 +00:00
model = model.Where("user_id = ?", in.UserId)
2023-09-21 11:01:48 +00:00
}
2023-09-21 02:30:36 +00:00
2023-09-21 11:01:48 +00:00
if in.DeliveryMethod != 0 {
2023-10-07 10:36:59 +00:00
model = model.Where("delivery_method = ?", in.DeliveryMethod)
2023-09-21 02:30:36 +00:00
}
2023-09-21 11:01:48 +00:00
if in.OrderCycle != "" {
// 下单时间
switch in.OrderCycle {
case "within_one_month":
2023-10-08 08:54:48 +00:00
model = model.Where("ctime >?", time.Now().UTC().AddDate(0, -1, 0))
2023-09-21 11:01:48 +00:00
case "within_three_month":
2023-10-08 08:54:48 +00:00
model = model.Where("ctime >?", time.Now().UTC().AddDate(0, -3, 0))
2023-09-21 11:01:48 +00:00
case "within_six_month":
2023-10-08 08:54:48 +00:00
model = model.Where("ctime >?", time.Now().UTC().AddDate(0, -6, 0))
2023-09-21 11:01:48 +00:00
case "within_one_year":
2023-10-08 08:54:48 +00:00
model = model.Where("ctime >?", time.Now().UTC().AddDate(-1, 0, 0))
2023-09-21 11:01:48 +00:00
}
}
var count int64
2023-11-13 09:08:50 +00:00
var modelTotal = model
resultCount := modelTotal.Count(&count)
2023-09-21 11:01:48 +00:00
if resultCount.Error != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "order count failed, err: %v", err)
2023-09-21 11:01:48 +00:00
return nil, resultCount.Error
}
var orderDetailList []gmodel.OrderDetail
if count > 0 {
2023-10-11 02:16:45 +00:00
m := model.Scopes(handlers.Paginate(&in.CurrentPage, &in.PerPage)).Order("id desc")
2023-09-21 11:01:48 +00:00
result := m.Find(&orderList)
if result.Error != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "order list failed, err: %v", err)
2023-09-21 11:01:48 +00:00
return nil, result.Error
}
for _, order := range orderList {
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &order, 1)
2023-09-21 11:01:48 +00:00
if err != nil {
return nil, err
}
orderDetailList = append(orderDetailList, ress.OrderDetail)
}
2023-10-10 03:29:28 +00:00
}
if len(orderList) == 0 {
2023-09-21 11:01:48 +00:00
orderDetailList = make([]gmodel.OrderDetail, 0)
}
2023-10-10 03:29:28 +00:00
2023-11-13 09:08:50 +00:00
var pageCount int64 = 1
if count > in.PerPage {
var float64Count = float64(count)
var float64PerPage = float64(in.PerPage)
pageCountFloat := math.Ceil(float64Count / float64PerPage)
pageCount = int64(pageCountFloat)
}
2023-09-21 11:01:48 +00:00
return &ListRes{
OrderDetailList: orderDetailList,
Meta: map[string]int64{
"total_count": count,
2023-11-13 09:08:50 +00:00
"page_count": pageCount,
2023-09-21 11:01:48 +00:00
"current_page": in.CurrentPage,
"per_page": in.PerPage,
},
}, nil
}
// 详情
func (d *defaultOrder) Detail(ctx context.Context, in *DetailReq) (res *DetailRes, err error) {
var errorCode basic.StatusResponse
2023-09-27 09:23:58 +00:00
var orderInfo gmodel.FsOrder
2023-09-21 11:01:48 +00:00
model := d.MysqlConn.Where("is_del = ?", 0)
if in.UserId != 0 {
2023-10-07 10:36:59 +00:00
model = model.Where("user_id = ?", in.UserId)
2023-09-21 11:01:48 +00:00
}
if in.OrderSn != "" {
2023-10-07 10:36:59 +00:00
model = model.Where("order_sn = ?", in.OrderSn)
2023-09-21 11:01:48 +00:00
}
2023-09-27 09:23:58 +00:00
result := model.Take(&orderInfo)
2023-09-21 11:01:48 +00:00
if result.Error != nil {
if errors.Is(result.Error, gorm.ErrRecordNotFound) {
errorCode = *basic.CodeErrOrderCreatePrePaymentInfoNoFound
} else {
errorCode = *basic.CodeServiceErr
}
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "order detail failed, err: %v", err)
2023-09-21 11:01:48 +00:00
return &DetailRes{
ErrorCode: errorCode,
}, result.Error
}
// 是否超时支付
2023-09-27 03:10:31 +00:00
// if *order.Status == int64(constants.ORDER_STATUS_UNPAIDDEPOSIT) {
// ctime := *order.Ctime
// ctimeTimeOut := ctime.Add(30 * time.Minute).UTC().Unix()
// ntimeTimeOut := time.Now().UTC().Unix()
// if ctimeTimeOut < ntimeTimeOut {
// errorCode = *basic.CodeErrOrderCreatePrePaymentTimeout
// err = errors.New("order pay timeout")
// logc.Errorf(ctx, "order detail failed, err: %v", err)
// return &DetailRes{
// ErrorCode: errorCode,
// }, err
// }
// }
2023-09-21 11:01:48 +00:00
2023-11-02 07:28:30 +00:00
ress, err := d.DetailHandler(ctx, &orderInfo, 1)
2023-09-21 11:01:48 +00:00
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "order detail failed, err: %v", err)
2023-09-21 11:01:48 +00:00
errorCode = *basic.CodeServiceErr
return &DetailRes{
ErrorCode: errorCode,
}, err
2023-09-21 04:11:15 +00:00
}
2023-09-21 06:26:14 +00:00
return &DetailRes{
2023-09-21 11:01:48 +00:00
ErrorCode: errorCode,
OrderDetail: ress.OrderDetail,
2023-09-21 06:26:14 +00:00
}, nil
2023-09-21 02:30:36 +00:00
}
2023-09-15 09:58:45 +00:00
// 下单
func (d *defaultOrder) Create(ctx context.Context, in *CreateReq) (res *CreateRes, err error) {
var errorCode basic.StatusResponse
2023-09-20 03:39:56 +00:00
// 订单编号
2023-09-21 02:30:36 +00:00
var orderSn string = order.GenerateOrderNumber()
2023-09-20 03:39:56 +00:00
2023-09-15 09:58:45 +00:00
err = d.MysqlConn.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
// 查询购物车
var shoppingCartList []*gmodel.RelaFsShoppingCart
2023-09-20 10:04:33 +00:00
resShoppingCartFind := tx.Table(gmodel.NewFsShoppingCartModel(tx).TableName()).
Preload("ShoppingCartProduct", func(dbPreload *gorm.DB) *gorm.DB {
return dbPreload.Table(gmodel.NewFsProductModel(tx).TableName()).Preload("CoverResource")
2023-09-26 08:16:58 +00:00
}).
2023-09-26 08:53:34 +00:00
Preload("ShoppingCartProductModel3d").
2023-09-20 07:10:18 +00:00
Preload("ShoppingCartProductModel3dFitting").
2023-09-15 09:58:45 +00:00
Where("id IN ?", in.CartIds).
Where("user_id = ?", in.UserId).
Find(&shoppingCartList)
err = resShoppingCartFind.Error
if err != nil {
return err
}
shoppingCartListLen := len(shoppingCartList)
if shoppingCartListLen == 0 {
errorCode = *basic.CodeErrOrderCreatShoppingCartEmpty
return errors.New(errorCode.Message)
}
if shoppingCartListLen != len(in.CartIds) {
errorCode = *basic.CodeErrOrderCreatShoppingCartNotMatched
return errors.New(errorCode.Message)
}
2023-09-18 07:39:12 +00:00
// 订单商品列表
2023-09-27 09:23:58 +00:00
var orderProductList []*gmodel.OrderProductInter
var shoppingProductSnapshotList []*gmodel.RelaFsProduct
var shoppingCartSnapshotList []*gmodel.FsShoppingCart
2023-09-15 09:58:45 +00:00
2023-09-18 07:39:12 +00:00
var shippingFee gmodel.AmountInfo
// 订单税费总价(厘)
var shippingFeeTotal int64
var tax = gmodel.AmountInfo{}
// 订单邮费总价(厘)
var taxTotal int64
var discount gmodel.AmountInfo
// 订单折扣总价(厘)
var discountTotal int64
var subtotal gmodel.AmountInfo
// 订单商品总价(厘)
var orderProductTotal int64
var total gmodel.AmountInfo
// 订单总价(厘)
var orderTotal int64
2023-09-19 11:17:04 +00:00
var nowTime = time.Now().UTC()
// 收货地址
var orderAddress *gmodel.OrderAddress
// 支付状态
2023-09-27 09:23:58 +00:00
var payStatus = constants.ORDER_PAY_STATUS_UNPAIDDEPOSIT
// 订单状态
var status = constants.ORDER_STATUS_UNPAIDDEPOSIT
2023-09-19 11:17:04 +00:00
// 直邮
if in.DeliveryMethod == constants.DELIVERYMETHODDIRECTMAIL {
orderAddress = &gmodel.OrderAddress{
Street: in.DeliveryAddress.Street,
City: in.DeliveryAddress.City,
FirstName: in.DeliveryAddress.FirstName,
LastName: in.DeliveryAddress.LastName,
Mobile: in.DeliveryAddress.Mobile,
State: in.DeliveryAddress.State,
Suite: in.DeliveryAddress.Suite,
ZipCode: in.DeliveryAddress.ZipCode,
2023-09-19 11:17:04 +00:00
}
}
2023-10-10 09:17:28 +00:00
// 预计交付时间
2023-10-10 09:30:31 +00:00
var expectedDeliveryTime = &gmodel.ExpectedDelivery{
2023-10-10 09:17:28 +00:00
Current: in.ExpectedDeliveryTime,
Initiate: in.ExpectedDeliveryTime,
}
2023-09-15 09:58:45 +00:00
for _, shoppingCart := range shoppingCartList {
// 购物车快照
2023-09-20 07:28:57 +00:00
var shoppingCartSnapshot gmodel.CartSnapshot
2023-09-15 09:58:45 +00:00
if shoppingCart.Snapshot != nil {
json.Unmarshal([]byte(*shoppingCart.Snapshot), &shoppingCartSnapshot)
}
// 商品异常
if shoppingCart.ShoppingCartProduct == nil || (shoppingCart.ShoppingCartProduct != nil && *shoppingCart.ShoppingCartProduct.IsShelf == 0) {
errorCode = *basic.CodeErrOrderCreatProductAbsent
2023-09-15 10:30:45 +00:00
errorCode.Message = "create order failed, product '" + shoppingCartSnapshot.ProductInfo.ProductName + "'is absent"
2023-09-15 09:58:45 +00:00
return errors.New(errorCode.Message)
}
2023-09-26 08:16:58 +00:00
var stepPriceJson gmodel.StepPriceJsonStruct
2023-09-26 08:53:34 +00:00
if shoppingCart.ShoppingCartProductModel3d.StepPrice != nil {
err = json.Unmarshal(*shoppingCart.ShoppingCartProductModel3d.StepPrice, &stepPriceJson)
2023-09-26 08:30:51 +00:00
if err != nil {
return err
}
2023-09-15 10:30:45 +00:00
} else {
errorCode = *basic.CodeErrOrderCreatProductPriceAbsent
errorCode.Message = "create order failed, step price of product '" + shoppingCartSnapshot.ProductInfo.ProductName + "'is failed"
2023-09-26 08:30:51 +00:00
return errors.New("shoppingCartProductModel3d.StepPrice nil")
2023-09-15 10:30:45 +00:00
}
2023-10-07 03:48:53 +00:00
var fittingPrice int64
if shoppingCart.ShoppingCartProductModel3dFitting != nil {
fittingPrice = *shoppingCart.ShoppingCartProductModel3dFitting.Price
}
2023-09-20 03:57:08 +00:00
/* 计算价格 */
2023-10-07 03:48:53 +00:00
productTotalPrice, productPrice, err := NewShoppingCart(tx, nil, nil).CaculateStepPrice(*shoppingCart.PurchaseQuantity, stepPriceJson, fittingPrice)
2023-09-20 03:55:51 +00:00
if err != nil {
errorCode = *basic.CodeErrOrderCreatProductPriceAbsent
2023-09-20 08:21:06 +00:00
errorCode.Message = "create order failed, step price of product '" + shoppingCartSnapshot.ProductInfo.ProductName + "'is failed"
return err
2023-09-20 03:55:51 +00:00
}
2023-09-20 03:57:08 +00:00
/* 计算价格 */
2023-09-18 07:39:12 +00:00
// 订单商品总价(厘)
orderProductTotal = orderProductTotal + productTotalPrice
// 订单商品
2023-09-20 07:07:12 +00:00
var productCoverMetadata map[string]interface{}
if shoppingCart.ShoppingCartProduct.CoverResource != nil && shoppingCart.ShoppingCartProduct.CoverResource.Metadata != nil {
json.Unmarshal(*shoppingCart.ShoppingCartProduct.CoverResource.Metadata, &productCoverMetadata)
}
2023-09-27 09:23:58 +00:00
productInter := gmodel.OrderProductInter{
2023-09-20 07:07:12 +00:00
TotalPrice: order.GetAmountInfo(order.GetAmountInfoReq{
2023-09-18 07:39:12 +00:00
ExchangeRate: in.ExchangeRate,
Initiate: productTotalPrice,
Current: productTotalPrice,
CurrentCurrency: in.CurrentCurrency,
OriginalCurrency: in.OriginalCurrency,
}),
2023-09-20 07:07:12 +00:00
ItemPrice: order.GetAmountInfo(order.GetAmountInfoReq{
2023-09-18 07:39:12 +00:00
ExchangeRate: in.ExchangeRate,
Initiate: productPrice,
Current: productPrice,
CurrentCurrency: in.CurrentCurrency,
OriginalCurrency: in.OriginalCurrency,
}),
2023-09-27 09:23:58 +00:00
PurchaseQuantity: &gmodel.PurchaseQuantity{
Current: *shoppingCart.PurchaseQuantity,
Initiate: *shoppingCart.PurchaseQuantity,
},
ProductId: *shoppingCart.ProductId,
ProductCover: *shoppingCart.ShoppingCartProduct.Cover,
ProductCoverMetadata: productCoverMetadata,
2023-11-01 10:13:01 +00:00
ProductName: shoppingCartSnapshot.ProductInfo.ProductName,
ProductSn: shoppingCartSnapshot.ProductInfo.ProductSn,
2023-11-08 10:17:00 +00:00
ProductWebsiteUnit: &shoppingCartSnapshot.ProductInfo.ProductWebsiteUnit,
2023-09-27 09:23:58 +00:00
DiyInformation: &shoppingCartSnapshot.UserDiyInformation,
2023-09-20 08:37:37 +00:00
FittingInfo: &gmodel.OrderProductFittingInfo{
FittingID: *shoppingCart.FittingId,
FittingName: shoppingCartSnapshot.FittingInfo.FittingName,
},
SizeInfo: &gmodel.OrderProductSizeInfo{
SizeID: *shoppingCart.SizeId,
Capacity: shoppingCartSnapshot.SizeInfo.Capacity,
Title: gmodel.OrderProductSizeInfoTitle{
Inch: shoppingCartSnapshot.SizeInfo.Inch,
Cm: shoppingCartSnapshot.SizeInfo.Cm,
},
},
2023-10-10 09:17:28 +00:00
IsHighlyCustomized: *shoppingCart.IsHighlyCustomized,
RenderImage: shoppingCartSnapshot.RenderImage,
CartId: shoppingCart.Id,
2023-10-10 09:30:31 +00:00
ExpectedDeliveryTime: expectedDeliveryTime,
2023-09-26 07:14:08 +00:00
}
2023-09-27 09:23:58 +00:00
orderProductList = append(orderProductList, &productInter)
shoppingProductSnapshotList = append(shoppingProductSnapshotList, shoppingCart.ShoppingCartProduct)
shoppingCartSnapshotList = append(shoppingCartSnapshotList, &shoppingCart.FsShoppingCart)
2023-09-15 09:58:45 +00:00
}
2023-09-18 07:39:12 +00:00
subtotal = order.GetAmountInfo(order.GetAmountInfoReq{
ExchangeRate: in.ExchangeRate,
Initiate: orderProductTotal,
Current: orderProductTotal,
CurrentCurrency: in.CurrentCurrency,
OriginalCurrency: in.OriginalCurrency,
})
orderTotal = orderProductTotal + shippingFeeTotal + taxTotal - discountTotal
total = order.GetAmountInfo(order.GetAmountInfoReq{
ExchangeRate: in.ExchangeRate,
Initiate: orderTotal,
Current: orderTotal,
CurrentCurrency: in.CurrentCurrency,
OriginalCurrency: in.OriginalCurrency,
})
2023-09-19 11:17:04 +00:00
// 定金
2023-09-20 03:39:56 +00:00
var depositInt int64 = orderTotal / 2
2023-09-19 11:17:04 +00:00
var deposit = gmodel.PayInfo{
Status: gmodel.PayStatus{
2023-09-27 09:23:58 +00:00
StatusCode: int64(constants.PAY_STATUS_UNPAID),
StatusTitle: constants.PayStatusMessage[constants.PAY_STATUS_UNPAID],
2023-09-19 11:17:04 +00:00
},
2023-09-22 08:27:17 +00:00
StatusLink: make([]gmodel.PayStatus, 0),
2023-09-20 03:39:56 +00:00
PayAmount: order.GetAmountInfo(order.GetAmountInfoReq{
ExchangeRate: in.ExchangeRate,
Initiate: depositInt,
Current: depositInt,
CurrentCurrency: in.CurrentCurrency,
OriginalCurrency: in.OriginalCurrency,
}),
2023-09-19 11:17:04 +00:00
}
2023-09-18 07:39:12 +00:00
2023-09-19 11:17:04 +00:00
// 尾款
2023-09-20 03:39:56 +00:00
var remainingBalanceInt int64 = orderTotal - depositInt
var remainingBalance = gmodel.PayInfo{
Status: gmodel.PayStatus{
2023-09-27 09:23:58 +00:00
StatusCode: int64(constants.PAY_STATUS_UNPAID),
StatusTitle: constants.PayStatusMessage[constants.PAY_STATUS_UNPAID],
2023-09-20 03:39:56 +00:00
},
2023-09-22 08:27:17 +00:00
StatusLink: make([]gmodel.PayStatus, 0),
2023-09-20 03:39:56 +00:00
PayAmount: order.GetAmountInfo(order.GetAmountInfoReq{
ExchangeRate: in.ExchangeRate,
Initiate: remainingBalanceInt,
Current: remainingBalanceInt,
CurrentCurrency: in.CurrentCurrency,
OriginalCurrency: in.OriginalCurrency,
}),
}
2023-09-19 11:17:04 +00:00
var orderAmount = gmodel.OrderAmount{
Deposit: deposit,
RemainingBalance: remainingBalance,
2023-09-20 03:39:56 +00:00
Discount: discount,
ShippingFee: shippingFee,
Tax: tax,
Subtotal: subtotal,
Total: total,
}
// 订单状态--链路
2023-09-20 07:07:12 +00:00
var statusLink = order.GenerateOrderStatusLink(in.DeliveryMethod, nowTime, in.ExpectedDeliveryTime)
2023-09-20 03:39:56 +00:00
2023-09-27 09:23:58 +00:00
byteOrderProduct, _ := json.Marshal(orderProductList)
2023-10-16 09:23:36 +00:00
byteOrderAddress, _ := json.Marshal(orderAddress)
2023-09-27 09:23:58 +00:00
byteOrderAmount, _ := json.Marshal(orderAmount)
byteShoppingCartSnapshot, _ := json.Marshal(shoppingCartSnapshotList)
byteShoppingProductSnapshot, _ := json.Marshal(shoppingProductSnapshotList)
byteStatusLink, _ := json.Marshal(statusLink)
2023-10-07 05:33:40 +00:00
byteOrderMetadata, _ := json.Marshal(gmodel.OrderMetadata{
2023-10-10 09:30:31 +00:00
ExpectedDeliveryTime: expectedDeliveryTime,
2023-10-07 05:33:40 +00:00
})
2023-09-20 10:04:33 +00:00
2023-09-27 02:09:26 +00:00
// 创建订单
2023-09-27 09:23:58 +00:00
resultCreate := tx.Create(&gmodel.FsOrder{
OrderProduct: &byteOrderProduct,
OrderAddress: &byteOrderAddress,
OrderAmount: &byteOrderAmount,
ShoppingCartSnapshot: &byteShoppingCartSnapshot,
ShoppingProductSnapshot: &byteShoppingProductSnapshot,
StatusLink: &byteStatusLink,
UserId: &in.UserId,
DeliveryMethod: &in.DeliveryMethod,
OrderSn: &orderSn,
Status: (*int64)(&status),
PayStatus: (*int64)(&payStatus),
Ctime: &nowTime,
2023-10-07 05:33:40 +00:00
Metadata: &byteOrderMetadata,
2023-09-27 09:23:58 +00:00
})
2023-09-27 02:09:26 +00:00
if resultCreate.Error != nil {
return resultCreate.Error
2023-09-19 11:17:04 +00:00
}
2023-09-27 10:00:35 +00:00
// 删除购物车
resultDelete := tx.Delete(&gmodel.FsShoppingCart{}, in.CartIds)
if resultDelete.Error != nil {
return resultDelete.Error
}
2023-09-27 02:09:26 +00:00
2023-09-15 09:58:45 +00:00
return nil
})
if err != nil {
2023-09-27 02:09:26 +00:00
logc.Errorf(ctx, "order create order failed, err: %v", err)
2023-09-20 10:04:33 +00:00
if errorCode.Code == 0 {
errorCode.Code = basic.CodeApiErr.Code
errorCode.Message = basic.CodeApiErr.Message
}
2023-09-20 03:39:56 +00:00
return &CreateRes{
OrderSn: orderSn,
ErrorCode: errorCode,
}, err
2023-09-15 09:58:45 +00:00
}
return &CreateRes{
2023-09-20 03:39:56 +00:00
OrderSn: orderSn,
2023-09-15 09:58:45 +00:00
ErrorCode: errorCode,
}, nil
}
2023-09-21 11:01:48 +00:00
// 详情处理
2023-11-02 07:28:30 +00:00
func (d *defaultOrder) DetailHandler(ctx context.Context, orderInfo *gmodel.FsOrder, original int64) (res *DetailRes, err error) {
2023-09-21 11:01:48 +00:00
2023-09-27 09:23:58 +00:00
var orderAmount gmodel.OrderAmount
err = json.Unmarshal(*orderInfo.OrderAmount, &orderAmount)
2023-09-21 11:01:48 +00:00
if err != nil {
2023-09-27 09:23:58 +00:00
logc.Errorf(ctx, "order detail handler unmarshal OrderAmount failed, err: %v", err)
2023-09-21 11:01:48 +00:00
return nil, err
}
2023-09-27 09:23:58 +00:00
var orderAddress *gmodel.OrderAddress
if orderInfo.OrderAddress != nil {
err = json.Unmarshal(*orderInfo.OrderAddress, &orderAddress)
if err != nil {
logc.Errorf(ctx, "order detail handler unmarshal OrderAddress failed, err: %v", err)
return nil, err
2023-09-22 08:27:17 +00:00
}
2023-09-27 09:23:58 +00:00
}
var orderProduct []gmodel.OrderProductInter
err = json.Unmarshal(*orderInfo.OrderProduct, &orderProduct)
if err != nil {
logc.Errorf(ctx, "order detail handler unmarshal OrderProduct failed, err: %v", err)
return nil, err
}
var shoppingCartSnapshot []gmodel.FsShoppingCart
err = json.Unmarshal(*orderInfo.ShoppingCartSnapshot, &shoppingCartSnapshot)
if err != nil {
logc.Errorf(ctx, "order detail handler unmarshal ShoppingCartSnapshot failed, err: %v", err)
return nil, err
}
var shoppingProductSnapshot []gmodel.RelaFsProduct
err = json.Unmarshal(*orderInfo.ShoppingProductSnapshot, &shoppingProductSnapshot)
if err != nil {
logc.Errorf(ctx, "order detail handler unmarshal ShoppingCartSnapshot failed, err: %v", err)
return nil, err
}
var statusLink []gmodel.OrderStatus
err = json.Unmarshal(*orderInfo.StatusLink, &statusLink)
if err != nil {
logc.Errorf(ctx, "order detail handler unmarshal StatusLink failed, err: %v", err)
return nil, err
}
var payStatusLink []gmodel.PayStatus
if orderInfo.PayStatusLink != nil {
err = json.Unmarshal(*orderInfo.PayStatusLink, &payStatusLink)
if err != nil {
logc.Errorf(ctx, "order detail handler unmarshal PayStatusLink failed, err: %v", err)
return nil, err
}
}
2023-10-07 05:33:40 +00:00
var orderMetadata gmodel.OrderMetadata
2023-09-27 09:23:58 +00:00
if orderInfo.Metadata != nil {
json.Unmarshal(*orderInfo.Metadata, &orderMetadata)
}
2023-10-07 05:33:40 +00:00
// 预计到货时间
var expectedTime = orderMetadata.ExpectedDeliveryTime.Current
var status = order.GetOrderStatusCurrent(statusLink, constants.OrderStatusCode(*orderInfo.Status))
status.ExpectedTime = &expectedTime
var orderDetail gmodel.OrderDetail
2023-09-27 09:23:58 +00:00
if original == 1 {
orderDetail.DeliveryAddress = orderAddress
orderDetail.OrderAmount = orderAmount
2023-10-27 08:36:09 +00:00
orderDetail.OrderAmount.Deposit.PayAmount = order.GetAmountInfoFormat(&orderAmount.Deposit.PayAmount, true)
orderDetail.OrderAmount.RemainingBalance.PayAmount = order.GetAmountInfoFormat(&orderAmount.RemainingBalance.PayAmount, true)
orderDetail.OrderAmount.Subtotal = order.GetAmountInfoFormat(&orderAmount.Subtotal, true)
orderDetail.OrderAmount.Total = order.GetAmountInfoFormat(&orderAmount.Total, true)
2023-09-27 09:23:58 +00:00
orderDetail.PayStatus = constants.OrderPayStatusCode(*orderInfo.PayStatus)
if *orderInfo.PayStatus == int64(constants.ORDER_STATUS_UNPAIDDEPOSIT) {
orderDetail.PayTimeout = time.Duration(orderInfo.Ctime.Add(30*time.Minute).UTC().Unix() - time.Now().UTC().Unix())
}
2023-09-27 09:23:58 +00:00
orderDetail.OrderInfo = gmodel.OrderInfo{
UserId: *orderInfo.UserId,
DeliveryMethod: *orderInfo.DeliveryMethod,
OrderSn: *orderInfo.OrderSn,
Ctime: orderInfo.Ctime,
Utime: orderInfo.Utime,
Status: status,
2023-09-27 09:40:44 +00:00
StatusLink: order.GetOrderStatusLinkUser(*orderInfo.DeliveryMethod, statusLink),
2023-09-27 09:23:58 +00:00
Metadata: orderMetadata,
}
2023-11-02 06:53:28 +00:00
// 购物车对应商品
2023-11-02 07:00:20 +00:00
var shoppingCartSnapshotDataMap = make(map[int64]gmodel.FsShoppingCartData, len(shoppingCartSnapshot))
2023-11-02 06:53:28 +00:00
for _, shoppingCartSnapshotValue := range shoppingCartSnapshot {
2023-11-02 07:16:44 +00:00
var snapshot map[string]interface{}
err = json.Unmarshal([]byte(*shoppingCartSnapshotValue.Snapshot), &snapshot)
2023-11-02 06:53:28 +00:00
if err != nil {
logc.Errorf(ctx, "OrderDetailHandler OrderShoppingCartSnapshotHandler failed, err: %v", err)
return nil, err
}
2023-11-02 07:16:44 +00:00
shoppingCartSnapshotDataMap[shoppingCartSnapshotValue.Id] = gmodel.FsShoppingCartData{
Id: shoppingCartSnapshotValue.Id,
UserId: shoppingCartSnapshotValue.UserId,
ProductId: shoppingCartSnapshotValue.ProductId,
TemplateId: shoppingCartSnapshotValue.TemplateId,
ModelId: shoppingCartSnapshotValue.ModelId,
SizeId: shoppingCartSnapshotValue.SizeId,
LightId: shoppingCartSnapshotValue.LightId,
FittingId: shoppingCartSnapshotValue.FittingId,
PurchaseQuantity: shoppingCartSnapshotValue.PurchaseQuantity,
IsSelected: shoppingCartSnapshotValue.IsSelected,
IsHighlyCustomized: shoppingCartSnapshotValue.IsHighlyCustomized,
Ctime: shoppingCartSnapshotValue.Ctime,
Utime: shoppingCartSnapshotValue.Utime,
Snapshot: &snapshot,
}
2023-11-02 06:53:28 +00:00
}
2023-09-27 09:23:58 +00:00
var orderProductList []gmodel.OrderProduct
for _, productValue := range orderProduct {
2023-10-07 09:31:24 +00:00
var selectColorIndex int64
2023-11-02 06:53:28 +00:00
var shoppingCartSnapshotData = shoppingCartSnapshotDataMap[productValue.CartId]
2023-11-02 07:16:44 +00:00
if shoppingCartSnapshotData.Snapshot != nil {
var snapshotInfoMap = *shoppingCartSnapshotData.Snapshot
2023-11-02 06:53:28 +00:00
selectColorIndexV, selectColorIndexE := snapshotInfoMap["select_color_index"]
if selectColorIndexE {
selectColorIndex = selectColorIndexV.(int64)
2023-09-27 09:23:58 +00:00
}
}
2023-11-02 06:53:28 +00:00
2023-10-11 02:14:31 +00:00
var expectedDeliveryTime time.Time
if productValue.ExpectedDeliveryTime == nil {
expectedDeliveryTime = expectedTime
} else {
expectedDeliveryTime = productValue.ExpectedDeliveryTime.Current
}
2023-09-27 09:23:58 +00:00
orderProductItem := gmodel.OrderProduct{
2023-10-27 08:36:09 +00:00
TotalPrice: order.GetAmountInfoFormat(&productValue.TotalPrice, true),
ItemPrice: order.GetAmountInfoFormat(&productValue.ItemPrice, true),
2023-10-10 09:17:28 +00:00
2023-09-27 09:23:58 +00:00
ProductId: productValue.ProductId,
ProductSn: productValue.ProductSn,
2023-11-01 10:13:01 +00:00
ProductWebsiteUnit: productValue.ProductWebsiteUnit,
2023-09-27 09:23:58 +00:00
ProductName: productValue.ProductName,
ProductCover: productValue.ProductCover,
ProductCoverMetadata: productValue.ProductCoverMetadata,
2023-11-01 10:13:01 +00:00
2023-09-27 09:23:58 +00:00
ShoppingCartSnapshot: &shoppingCartSnapshotData,
2023-10-11 02:14:31 +00:00
ExpectedDeliveryTime: &expectedDeliveryTime,
2023-10-27 08:36:09 +00:00
PurchaseQuantity: order.GetPurchaseQuantity(productValue.PurchaseQuantity, true),
2023-09-27 09:23:58 +00:00
DiyInformation: productValue.DiyInformation,
SizeInfo: productValue.SizeInfo,
FittingInfo: productValue.FittingInfo,
IsHighlyCustomized: productValue.IsHighlyCustomized,
2023-10-07 05:33:40 +00:00
RenderImage: productValue.RenderImage,
2023-10-07 09:31:24 +00:00
SelectColorIndex: selectColorIndex,
2023-09-27 09:23:58 +00:00
}
orderProductList = append(orderProductList, orderProductItem)
}
orderDetail.OrderProduct = orderProductList
2023-09-22 08:27:17 +00:00
}
2023-09-21 11:01:48 +00:00
return &DetailRes{
2023-09-27 09:23:58 +00:00
OrderDetail: orderDetail,
OrderDetailOriginal: OrderDetailOriginal{
Status: &status,
OrderAmount: &orderAmount,
OrderAddress: orderAddress,
OrderProduct: orderProduct,
ShoppingCartSnapshot: shoppingCartSnapshot,
ShoppingProductSnapshot: shoppingProductSnapshot,
StatusLink: statusLink,
PayStatusLink: payStatusLink,
},
2023-09-21 11:01:48 +00:00
}, nil
}