833 lines
34 KiB
Go
833 lines
34 KiB
Go
package tao_vegetable
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"git.rosy.net.cn/baseapi/platformapi/tao_vegetable"
|
|
domain3156 "git.rosy.net.cn/baseapi/platformapi/tao_vegetable/sdk/ability3156/domain"
|
|
request3156 "git.rosy.net.cn/baseapi/platformapi/tao_vegetable/sdk/ability3156/request"
|
|
domain591 "git.rosy.net.cn/baseapi/platformapi/tao_vegetable/sdk/ability591/domain"
|
|
request591 "git.rosy.net.cn/baseapi/platformapi/tao_vegetable/sdk/ability591/request"
|
|
"git.rosy.net.cn/baseapi/platformapi/tao_vegetable/sdk/util"
|
|
"git.rosy.net.cn/baseapi/utils"
|
|
"git.rosy.net.cn/jx-callback/business/jxutils"
|
|
"git.rosy.net.cn/jx-callback/business/jxutils/jxcontext"
|
|
"git.rosy.net.cn/jx-callback/business/jxutils/tasksch"
|
|
"git.rosy.net.cn/jx-callback/business/model"
|
|
"git.rosy.net.cn/jx-callback/business/model/dao"
|
|
"git.rosy.net.cn/jx-callback/business/partner"
|
|
"git.rosy.net.cn/jx-callback/globals"
|
|
)
|
|
|
|
var (
|
|
specPat = regexp.MustCompile(`(\d+)(.+)`)
|
|
)
|
|
|
|
var (
|
|
VendorStatus2StatusMap = map[string]int{
|
|
tao_vegetable.OrderStatusPayFinsh: model.OrderStatusNew, // 商户接单
|
|
tao_vegetable.OrderStatusNew: model.OrderStatusAccepted, // 商户接单
|
|
tao_vegetable.OrderStatusPickedUp: model.OrderStatusFinishedPickup, // 拣货完成
|
|
tao_vegetable.OrderStatusCallRider: model.OrderStatusFinishedPickup, // 打包出库(呼叫骑手,骑手到店,骑手取货)
|
|
tao_vegetable.OrderStatusDelivery: model.OrderStatusDelivering, // 配送中
|
|
tao_vegetable.OrderStatusDeliveryOver: model.OrderStatusFinished, // 配送结束
|
|
tao_vegetable.OrderStatusUserRejection: model.OrderStatusDeliverFailed, // 用户拒收
|
|
tao_vegetable.OrderStatusMerchantCancel: model.OrderStatusCanceled, // 商户取消订单
|
|
tao_vegetable.OrderStatusSuccess: model.OrderStatusFinished, // 订单完成
|
|
}
|
|
)
|
|
|
|
func (p *PurchaseHandler) getStatusFromVendorStatus(vendorStatus string) int {
|
|
if status, ok := VendorStatus2StatusMap[vendorStatus]; ok {
|
|
return status
|
|
}
|
|
return model.OrderStatusUnknown
|
|
}
|
|
|
|
// getOrder 获取订单详情
|
|
func (p *PurchaseHandler) getOrder(vendorOrgCode string, vendorOrderID int64, vendorStoreID string) (order *model.GoodsOrder, orderMap *domain591.AlibabaAelophyOrderGetOrderResponse, err error) {
|
|
requestParam := &request591.AlibabaAelophyOrderGetRequest{OrderGetRequest: &domain591.AlibabaAelophyOrderGetOrderGetRequest{
|
|
StoreId: utils.String2Pointer(vendorStoreID),
|
|
BizOrderId: utils.Int64ToPointer(vendorOrderID),
|
|
}}
|
|
|
|
orderDetail, err := getAPI(vendorOrgCode, 0, vendorStoreID).QueryOrderDetail(requestParam)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
orderMap = orderDetail
|
|
order = &model.GoodsOrder{
|
|
VendorOrderID: utils.Int64ToStr(vendorOrderID),
|
|
VendorID: model.VendorIDTaoVegetable,
|
|
VendorStoreID: vendorStoreID,
|
|
VendorOrderID2: *orderDetail.OutOrderId,
|
|
StoreID: utils.Str2Int(*orderDetail.StoreId),
|
|
JxStoreID: utils.Str2Int(*orderDetail.StoreId),
|
|
CoordinateType: model.CoordinateTypeMars,
|
|
BuyerComment: *orderDetail.ReceiveInfo.ReceiverMemo,
|
|
PickDeadline: utils.DefaultTimeValue,
|
|
VendorStatus: *orderDetail.OrderStatus, // PAID = 订单支付完成 PACKAGED = 订单打包出库 SHIPPING = 订单配送揽收 SUCCESS = 交易完成 CLOSE = 订单取消
|
|
StatusTime: time.Now(),
|
|
OrderCreatedAt: utils.Str2TimeWithDefault(orderDetail.PayTime.String(), time.Now()),
|
|
OriginalData: string(utils.MustMarshal(orderDetail)),
|
|
ActualPayPrice: *orderDetail.PayFee,
|
|
BaseFreightMoney: *orderDetail.PostFee,
|
|
InvoiceTitle: "",
|
|
InvoiceTaxerID: "",
|
|
InvoiceEmail: "",
|
|
VendorOrgCode: vendorOrgCode,
|
|
UserID: *orderDetail.OpenUid,
|
|
}
|
|
|
|
if *orderDetail.OrderStatus == tao_vegetable.OrderStatusDeliveryOver {
|
|
order.OrderFinishedAt = time.Now()
|
|
} else {
|
|
order.OrderFinishedAt = utils.DefaultTimeValue
|
|
}
|
|
|
|
order.Status = p.getStatusFromVendorStatus(*orderDetail.OrderStatus)
|
|
originalList := strings.Split(*orderDetail.ReceiveInfo.ReceiverPoi, ",")
|
|
order.ConsigneeLng = jxutils.StandardCoordinate2Int(utils.Str2Float64(originalList[0]))
|
|
order.ConsigneeLat = jxutils.StandardCoordinate2Int(utils.Str2Float64(originalList[1]))
|
|
order.DiscountMoney = *orderDetail.DiscountFee
|
|
|
|
//var salePrice int64 = 0
|
|
//var weight int = 0
|
|
order.PmSubsidyMoney = *orderDetail.SkuDiscountPlatformFee // 平台承担优惠
|
|
// 添加需要赠送的东西(暂时没有赠品套餐直接商品)
|
|
multiSkuMap := make(map[int]int)
|
|
if len(*orderDetail.SubOrderResponseList) > 0 {
|
|
for _, extra := range *orderDetail.SubOrderResponseList {
|
|
sku := &model.OrderSku{
|
|
VendorOrderID: order.VendorOrderID,
|
|
VendorSubOrderID: *extra.OutSubOrderId,
|
|
VendorID: model.VendorIDTaoVegetable,
|
|
StoreSubID: 0,
|
|
StoreSubName: "",
|
|
Count: utils.Float64TwoInt(*extra.BuySaleQuantity),
|
|
VendorSkuID: *extra.SkuCode,
|
|
SkuID: utils.Str2Int(*extra.SkuCode),
|
|
JxSkuID: utils.Str2Int(*extra.SkuCode),
|
|
SkuName: *extra.SkuName,
|
|
ShopPrice: *extra.Price,
|
|
VendorPrice: *extra.OriginalFee / utils.Float64TwoInt64(*extra.BuySaleQuantity),
|
|
SalePrice: *extra.OriginalFee / utils.Float64TwoInt64(*extra.BuySaleQuantity),
|
|
EarningPrice: 0,
|
|
Weight: int(*extra.Weight / utils.Float64TwoInt64(*extra.BuySaleQuantity)),
|
|
SkuType: 0,
|
|
PromotionType: 0,
|
|
OrderCreatedAt: order.OrderCreatedAt,
|
|
IsVendorAct: 0,
|
|
Upc: *extra.Barcode,
|
|
}
|
|
|
|
activityId := make([]int64, 0)
|
|
activityName := make([]string, 0)
|
|
if extra.Activitys != nil {
|
|
for _, v := range *extra.Activitys {
|
|
// 渠道活动
|
|
if v.ChannelActivityId != nil {
|
|
activityId = append(activityId, utils.Str2Int64WithDefault(*v.ChannelActivityId, 999))
|
|
activityName = append(activityName, *v.ChannelActivityId+":"+*v.ActivityName)
|
|
}
|
|
// 业务活动
|
|
if v.BizActivityId != nil {
|
|
activityId = append(activityId, utils.Str2Int64WithDefault(*v.BizActivityId, 999))
|
|
activityName = append(activityName, *v.BizActivityId+":"+*v.ActivityName)
|
|
}
|
|
// 商家erp活动
|
|
if v.MerchantActivityId != nil {
|
|
activityId = append(activityId, utils.Str2Int64WithDefault(*v.MerchantActivityId, 999))
|
|
activityName = append(activityName, *v.MerchantActivityId+":"+*v.ActivityName)
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(activityId) > 0 {
|
|
sku.StoreSubID = int(activityId[0])
|
|
sku.StoreSubName = strings.Join(activityName, ",")
|
|
}
|
|
if sku.Weight == 0 {
|
|
sku.Weight = 222 // 如果名字里找不到缺省给半斤左右的一个特别值
|
|
}
|
|
multiSkuMap[sku.SkuID]++
|
|
order.Skus = append(order.Skus, sku)
|
|
//salePrice += sku.SalePrice
|
|
//weight += sku.Weight
|
|
globals.SugarLogger.Debugf("=====skuPrice := %d", sku.SalePrice)
|
|
}
|
|
}
|
|
//order.SalePrice = salePrice
|
|
//order.Weight = weight
|
|
//globals.SugarLogger.Debugf("=====skuPrice := %d", salePrice)
|
|
|
|
// 淘宝默认自配送
|
|
if *orderDetail.DeliveryType == tao_vegetable.OrderDeliveryTypeTime {
|
|
order.DeliveryType = model.OrderDeliveryTypePlatform
|
|
} else if *orderDetail.DeliveryType == tao_vegetable.OrderDeliveryTypeSelf {
|
|
order.DeliveryType = model.OrderDeliveryTypeSelfTake
|
|
} else if *orderDetail.DeliveryType == tao_vegetable.OrderDeliveryTypeStore {
|
|
order.DeliveryType = model.OrderDeliveryTypeStoreSelf
|
|
}
|
|
|
|
// 期望送达时间两小时内为立即达
|
|
earliestTime := utils.Str2Time(strings.Split(*orderDetail.ReceiveInfo.ExpectArriveTime, "~")[0]).Unix()
|
|
if earliestTime-time.Now().Unix() > 2*60*60 {
|
|
order.BusinessType = model.BusinessTypeImmediate
|
|
order.ExpectedDeliveredTime = getTimeFromTimestamp(earliestTime + 30*60) // 预计最晚送达时间
|
|
} else { // 定时达
|
|
order.BusinessType = model.BusinessTypeDingshida
|
|
order.ExpectedDeliveredTime = getTimeFromTimestamp(earliestTime + 30*60) // 预计最晚送达时间
|
|
}
|
|
|
|
// 用户信息
|
|
order.ConsigneeName = *orderDetail.ReceiveInfo.ReceiverName
|
|
order.ConsigneeMobile = *orderDetail.ReceiveInfo.ReceiverPhone
|
|
order.ConsigneeAddress = *orderDetail.ReceiveInfo.ReceiverAddress
|
|
for _, v := range order.Skus {
|
|
if multiSkuMap[v.SkuID] > 1 && v.SalePrice == v.VendorPrice {
|
|
v.IsVendorAct = model.YES
|
|
}
|
|
}
|
|
// 抖音订单手机号和收货地址是否同城(虚拟号,无法获取到正确地址)
|
|
order.PhoneAscription = model.PhoneAscriptionAddressNo + "-" + "归属信息不匹配:" + "虚拟电话号码"
|
|
//ascription, err := ascription_place.Find(order.ConsigneeMobile)
|
|
//if err != nil {
|
|
// order.PhoneAscription = model.PhoneAscriptionAddressNo + "-" + err.Error()
|
|
// err = nil
|
|
//} else {
|
|
// if strings.Contains(order.ConsigneeAddress, ascription.Province) && strings.Contains(order.ConsigneeAddress, ascription.City) {
|
|
// order.PhoneAscription = model.PhoneAscriptionAddressYes + "-" + ascription.Province + ascription.City
|
|
// } else {
|
|
// order.PhoneAscription = model.PhoneAscriptionAddressNo + "-" + "归属信息不匹配:" + ascription.Province + ascription.City
|
|
// }
|
|
//}
|
|
|
|
// 本地获取订单记录
|
|
orderSeq, _ := dao.GetVendorOrderNumber(dao.GetDB(), model.VendorIDTaoVegetable, order.VendorStoreID)
|
|
order.OrderSeq = orderSeq + 1
|
|
|
|
// 包装袋金额
|
|
store, _ := dao.GetStoreDetail(dao.GetDB(), order.JxStoreID, order.VendorID, order.VendorOrgCode)
|
|
if store != nil {
|
|
order.PackagePrice = int(*orderDetail.PackageFee) + store.PackageSetting
|
|
} else {
|
|
order.PackagePrice = int(*orderDetail.PackageFee)
|
|
}
|
|
order.StoreName = store.Name // 真实门店名称
|
|
|
|
globals.SugarLogger.Debugf("=====order : %s", utils.Format4Output(order, false))
|
|
return order, orderMap, err
|
|
}
|
|
|
|
type RiderInfo struct {
|
|
OrderId string `json:"order_id"` // 发单平台订单id(美团,京西,京,京东)
|
|
ThirdCarrierOrderId string `json:"third_carrier_order_id"` // 京西平台id(运单id)
|
|
CourierName string `json:"courier_name"` // 骑手名称
|
|
CourierPhone string `json:"courier_phone"` // 骑手电话
|
|
LogisticsProviderCode string `json:"logistics_provider_code"` // 配送平台code 10001-顺丰, 10002-达达, 10003-闪送, 10004-蜂鸟, 10005 UU跑腿,10006 快跑者, 10007 极客快送,10008-点我达,10009 同达, 10010-生活半径,10011 邻趣,10012 趣送, 10013 快服务 10014 菜鸟新配盟 10015 商家自建配送 10016 风先生,10017-其他,10018-抖音配送(小时达),10032-美团跑腿
|
|
LogisticsStatus int `json:"logistics_status"` // 配送状态(美团用)
|
|
LogisticsContext string `json:"logistics_context"` // 配送状态描述
|
|
Latitude string `json:"latitude"` // 骑手当前的纬度,美团使用的是高德坐标系。
|
|
Longitude string `json:"longitude"` // 骑手当前的经度,美团使用的是高德坐标系。
|
|
OpCode string `json:"opcode"` // 抖音状态(抖音才需要)
|
|
}
|
|
|
|
// GetOrderRider 自配送商家同步发货状态和配送信息(推荐)
|
|
func (p *PurchaseHandler) GetOrderRider(vendorOrgCode, vendorStoreID string, param map[string]interface{}) (err error) {
|
|
req := &request591.AlibabaAelophyOrderLogisticsTraceCallbackRequest{
|
|
LogisticsTraceCallbackRequest: &domain591.AlibabaAelophyOrderLogisticsTraceCallbackLogisticsTraceCallbackRequest{
|
|
StoreId: utils.String2Pointer(vendorStoreID),
|
|
BizOrderId: utils.Int64ToPointer(utils.Str2Int64(param["order_id"].(string))),
|
|
Longitude: utils.String2Pointer(param["longitude"].(string)),
|
|
Latitude: utils.String2Pointer(param["latitude"].(string)),
|
|
UpdateTime: (*util.LocalTime)(utils.Time2Pointer(time.Now())),
|
|
},
|
|
}
|
|
return getAPI(vendorOrgCode, 0, vendorStoreID).DeliveryTrajectory(req)
|
|
}
|
|
|
|
func (p *PurchaseHandler) GetOrder(vendorOrgCode, vendorOrderID, vendorStoreID string) (order *model.GoodsOrder, err error) {
|
|
order, _, err = p.getOrder(vendorOrgCode, utils.Str2Int64(vendorOrderID), vendorStoreID)
|
|
return order, err
|
|
}
|
|
|
|
func (p *PurchaseHandler) GetOrderStatus(vendorOrgCode, vendorOrderID string) (status int, err error) {
|
|
order, _ := partner.CurOrderManager.LoadOrder(vendorOrderID, model.VendorIDTaoVegetable)
|
|
if order == nil {
|
|
return 0, err
|
|
}
|
|
|
|
requestParam := &request591.AlibabaAelophyOrderGetRequest{OrderGetRequest: &domain591.AlibabaAelophyOrderGetOrderGetRequest{
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
BizOrderId: utils.Int64ToPointer(utils.Str2Int64(vendorOrderID)),
|
|
}}
|
|
orderDetail, err := getAPI(vendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").QueryOrderDetail(requestParam)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
status = p.getStatusFromVendorStatus(*orderDetail.OrderStatus)
|
|
return status, err
|
|
}
|
|
|
|
func (c *PurchaseHandler) Map2Order(orderData map[string]interface{}) (order *model.GoodsOrder) {
|
|
order, _, _ = c.getOrder(orderData["VendorOrgCode"].(string), utils.Str2Int64(orderData["vendorOrderID"].(string)), "")
|
|
return order
|
|
}
|
|
|
|
// getRefundSkuDetailList 获取商家部分退款的订单列表
|
|
func getRefundSkuDetailList(order *model.GoodsOrder) (skuList []*domain591.AlibabaAelophyOrderGetSubOrderResponse, err error) {
|
|
requestParam := &request591.AlibabaAelophyOrderGetRequest{OrderGetRequest: &domain591.AlibabaAelophyOrderGetOrderGetRequest{
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
BizOrderId: utils.Int64ToPointer(utils.Str2Int64(order.VendorOrderID)),
|
|
}}
|
|
orderDetail, err := getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").QueryOrderDetail(requestParam)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, v := range *orderDetail.SubOrderResponseList {
|
|
if *v.OrderStatus == tao_vegetable.OrderStatusRefundClose {
|
|
skuList = append(skuList, &v)
|
|
}
|
|
}
|
|
return skuList, err
|
|
}
|
|
|
|
func getSkuWeight(product map[string]interface{}) (weight int) {
|
|
if weight = int(utils.Interface2Int64WithDefault(product["weight"], 0)); weight == 0 {
|
|
searchResult := specPat.FindStringSubmatch(product["spec"].(string))
|
|
if len(searchResult) == 3 {
|
|
weight = jxutils.FormatSkuWeight(float32(utils.Str2Float64WithDefault(searchResult[1], 0)), utils.TrimBlankChar(searchResult[2]))
|
|
}
|
|
if weight == 0 {
|
|
_, _, _, specUnit, _, specQuality := jxutils.SplitSkuName(product["food_name"].(string))
|
|
weight = jxutils.FormatSkuWeight(specQuality, specUnit)
|
|
}
|
|
}
|
|
return weight
|
|
}
|
|
|
|
func (c *PurchaseHandler) onOrderMsg(orderStatus, orderId string, orderCallback interface{}) (response *tao_vegetable.CallBackResult) {
|
|
var err error
|
|
|
|
// 售后单
|
|
if c.isAfsMsg(orderStatus, orderId) {
|
|
response = c.OnAfsOrderMsg(orderId, orderStatus, orderCallback)
|
|
return response
|
|
}
|
|
|
|
// 正常订单
|
|
msg := orderCallback.(*tao_vegetable.CallbackOrder)
|
|
if orderStatus == tao_vegetable.OrderStatusOnSaleCancel {
|
|
onSale := orderCallback.(*tao_vegetable.OnSaleCancel)
|
|
//msg.PublicModel = onSale.PublicModel
|
|
msg.MerchantCode = onSale.PartCancelRequest.MerchantCode
|
|
msg.StoreId = onSale.PartCancelRequest.StoreId
|
|
msg.BizOrderId = onSale.PartCancelRequest.BizOrderId
|
|
msg.OrderStatus = tao_vegetable.OrderStatusOnSaleCancel
|
|
}
|
|
status := c.callbackOrderMsg2Status(msg)
|
|
// 校验重复消息
|
|
if partner.CurOrderManager.GetStatusDuplicatedCount(status) > 0 {
|
|
return tao_vegetable.CallBackResultInfo(nil)
|
|
}
|
|
|
|
// 商户接单/支付完成代表新订单
|
|
if msg.OrderStatus == tao_vegetable.OrderStatusPayFinsh {
|
|
order, orderMap, err2 := c.getOrder("", msg.BizOrderId, msg.StoreId)
|
|
if err = err2; err == nil {
|
|
err = partner.CurOrderManager.OnOrderNew(order, status)
|
|
if err == nil {
|
|
utils.CallFuncAsync(func() {
|
|
if msg.OrderStatus == tao_vegetable.OrderStatusPayFinsh {
|
|
c.OnOrderDetail(orderMap, partner.CreatedPeration)
|
|
} else {
|
|
c.OnOrderDetail(orderMap, partner.UpdatedPeration)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
} else {
|
|
order, err := partner.CurOrderManager.LoadOrder(utils.Int64ToStr(msg.BizOrderId), model.VendorIDTaoVegetable)
|
|
if err != nil {
|
|
return tao_vegetable.CallBackResultInfo(err)
|
|
}
|
|
if status.Status == model.OrderStatusAdjust {
|
|
//skuList, err2 := getRefundSkuDetailList(msg, order)
|
|
//if err = err2; err == nil {
|
|
// var removedSkuList []*model.OrderSku
|
|
// for _, mtwmSku := range skuList {
|
|
// order.ActualPayPrice -= jxutils.StandardPrice2Int(mtwmSku.RefundPrice) * int64(mtwmSku.Count)
|
|
// removedSkuList = append(removedSkuList, &model.OrderSku{
|
|
// SkuID: int(utils.Str2Int64WithDefault(mtwmSku.SkuID, 0)),
|
|
// Count: mtwmSku.Count,
|
|
// })
|
|
// }
|
|
// order = jxutils.RemoveSkuFromOrder(order, removedSkuList)
|
|
// jxutils.RefreshOrderSkuRelated(order)
|
|
// err = partner.CurOrderManager.OnOrderAdjust(order, status)
|
|
//}
|
|
} else {
|
|
// 发货完成
|
|
if msg.OrderStatus == tao_vegetable.OrderStatusCallRider { // || msgId == tiktokShop.CallbackPartGoodsMsgTagId 部分发货
|
|
utils.CallFuncAsync(func() {
|
|
orderMap, _ := getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").QueryOrderDetail(&request591.AlibabaAelophyOrderGetRequest{
|
|
OrderGetRequest: &domain591.AlibabaAelophyOrderGetOrderGetRequest{
|
|
StoreId: utils.String2Pointer(msg.StoreId),
|
|
BizOrderId: utils.Int64ToPointer(msg.BizOrderId),
|
|
},
|
|
})
|
|
c.OnOrderDetail(orderMap, partner.UpdatedPeration)
|
|
})
|
|
}
|
|
if err := partner.CurOrderManager.OnOrderStatusChanged(order.VendorOrgCode, status); err != nil {
|
|
return tao_vegetable.CallBackResultInfo(err)
|
|
}
|
|
|
|
}
|
|
}
|
|
return tao_vegetable.CallBackResultInfo(err)
|
|
}
|
|
|
|
func (c *PurchaseHandler) callbackOrderMsg2Status(msg *tao_vegetable.CallbackOrder) (orderStatus *model.OrderStatus) {
|
|
orderId := utils.Int64ToStr(msg.BizOrderId)
|
|
orderStatus = &model.OrderStatus{
|
|
VendorOrderID: orderId,
|
|
VendorID: model.VendorIDTaoVegetable,
|
|
OrderType: model.OrderTypeOrder,
|
|
RefVendorOrderID: orderId,
|
|
RefVendorID: model.VendorIDTaoVegetable,
|
|
VendorStatus: msg.OrderStatus,
|
|
//StatusTime: utils.Str2TimeWithDefault(msg.Timestamp, time.Now()),
|
|
StatusTime: time.Now(),
|
|
}
|
|
|
|
switch msg.OrderStatus {
|
|
case tao_vegetable.OrderStatusPayFinsh: // 新订单
|
|
orderStatus.Status = model.OrderStatusNew
|
|
orderStatus.Remark = "新订单"
|
|
case tao_vegetable.OrderStatusNew: // 商户接单
|
|
orderStatus.Status = model.OrderStatusAccepted
|
|
orderStatus.Remark = "商户接单"
|
|
case tao_vegetable.OrderStatusPickedUp: // 拣货完成
|
|
orderStatus.Status = model.OrderStatusFinishedPickup
|
|
orderStatus.Remark = "拣货完成"
|
|
case tao_vegetable.OrderStatusCallRider: // 打包出库(呼叫骑手,骑手到店,骑手取货)
|
|
orderStatus.Status = model.OrderStatusFinishedPickup
|
|
orderStatus.Remark = "打包出库(呼叫骑手,骑手到店,骑手取货)"
|
|
case tao_vegetable.OrderStatusDelivery: // 配送中
|
|
orderStatus.Status = model.OrderStatusDelivering
|
|
orderStatus.Remark = "配送中"
|
|
case tao_vegetable.OrderStatusDeliveryOver: // 配送结束
|
|
orderStatus.Status = model.OrderStatusFinished
|
|
orderStatus.Remark = "送达"
|
|
case tao_vegetable.OrderStatusUserRejection: // 用户拒收
|
|
orderStatus.Status = model.OrderStatusDeliverFailed
|
|
orderStatus.Remark = "用户拒收"
|
|
case tao_vegetable.OrderStatusMerchantCancel: // 商户取消订单
|
|
orderStatus.Status = model.OrderStatusCanceled
|
|
orderStatus.Remark = "商户取消"
|
|
case tao_vegetable.OrderStatusOnSaleCancel:
|
|
orderStatus.Status = model.OrderStatusCanceled
|
|
orderStatus.Remark = "用户售中取消"
|
|
case tao_vegetable.OrderStatusSuccess: // 送达
|
|
orderStatus.Status = model.OrderStatusFinished
|
|
orderStatus.Remark = "订单送达"
|
|
}
|
|
return orderStatus
|
|
}
|
|
|
|
func (c *PurchaseHandler) postFakeMsg(vendorOrderID, cmd, vendorStatus string) {
|
|
msg := &tao_vegetable.CallbackOrder{
|
|
//PublicModel: tao_vegetable.PublicModel{
|
|
// Method: "",
|
|
// AppKey: "",
|
|
// Session: "",
|
|
// Timestamp: utils.Time2Str(time.Now()),
|
|
// V: "",
|
|
// SignMethod: "",
|
|
// Sign: "",
|
|
// Format: "",
|
|
// Simplify: false,
|
|
// CustomerId: false,
|
|
//},
|
|
MerchantCode: "",
|
|
StoreId: "",
|
|
BizOrderId: utils.Str2Int64(vendorOrderID),
|
|
OrderStatus: vendorStatus,
|
|
}
|
|
utils.CallFuncAsync(func() {
|
|
c.onOrderMsg(vendorStatus, vendorOrderID, msg)
|
|
})
|
|
}
|
|
|
|
// AcceptOrRefuseOrder 自动接单
|
|
func (c *PurchaseHandler) AcceptOrRefuseOrder(order *model.GoodsOrder, isAcceptIt bool, userName string) (err error) {
|
|
if isAcceptIt {
|
|
param := &request591.AlibabaAelophyOrderWorkCallbackRequest{}
|
|
param.WorkCallbackRequest = &domain591.AlibabaAelophyOrderWorkCallbackWorkCallbackRequest{
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
BizOrderId: utils.Int64ToPointer(utils.Str2Int64(order.VendorOrderID)),
|
|
Status: utils.String2Pointer(tao_vegetable.OrderStatusNew),
|
|
StatusRemark: nil,
|
|
DelivererName: nil,
|
|
DelivererPhone: nil,
|
|
WorkCallbackSubOrderInfoList: nil,
|
|
DelivererCompany: nil,
|
|
LogisticsNo: nil,
|
|
}
|
|
err = getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), order.VendorStoreID).DeliveryFinish(param)
|
|
if err != nil {
|
|
globals.SugarLogger.Warnf("tao AcceptOrRefuseOrder orderID:%s failed with err:%v", order.VendorOrderID, err)
|
|
}
|
|
} else {
|
|
err = c.CancelOrder(jxcontext.AdminCtx, order, "bu")
|
|
}
|
|
return err
|
|
}
|
|
|
|
// PickupGoods 拣货
|
|
func (c *PurchaseHandler) PickupGoods(order *model.GoodsOrder, isSelfDelivery bool, userName string) (err error) {
|
|
globals.SugarLogger.Debugf("PickupGoods 自动拣货:%s, %s", utils.Format4Output(isSelfDelivery, false), utils.Format4Output(order, false))
|
|
if isSelfDelivery {
|
|
param, err := orderStatusChangeNotice(order, tao_vegetable.OrderStatusPickedUp)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err = getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").DeliveryFinish(param); err != nil {
|
|
globals.SugarLogger.Debugf("PickupGoods 拣货失败可能是BizSubOrderId 没填写 : %s", err.Error())
|
|
return err
|
|
}
|
|
}
|
|
|
|
c.postFakeMsg(order.VendorOrderID, tao_vegetable.OrderStatusPickedUp, tao_vegetable.OrderStatusPickedUp)
|
|
return err
|
|
}
|
|
|
|
func (p *PurchaseHandler) AcceptOrRefuseFailedGetOrder(ctx *jxcontext.Context, order *model.GoodsOrder, isAcceptIt bool) (err error) {
|
|
return err
|
|
}
|
|
|
|
func (p *PurchaseHandler) CallCourier(ctx *jxcontext.Context, order *model.GoodsOrder) (err error) { // 拣货失败后再次招唤平台配送
|
|
return err
|
|
}
|
|
|
|
func (p *PurchaseHandler) ConfirmReceiveGoods(ctx *jxcontext.Context, order *model.GoodsOrder) (err error) { // 投递失败后确认收到退货
|
|
return err
|
|
}
|
|
|
|
// CanSwitch2SelfDeliver 判断订单能不能转自送,淘宝为纯自送门店
|
|
func (c *PurchaseHandler) CanSwitch2SelfDeliver(order *model.GoodsOrder) (isCan bool, err error) {
|
|
return true, nil
|
|
}
|
|
|
|
// Swtich2SelfDeliver 转自送接口通知淘鲜达发货
|
|
func (c *PurchaseHandler) Swtich2SelfDeliver(order *model.GoodsOrder, userName string) (err error) {
|
|
param, err := orderStatusChangeNotice(order, tao_vegetable.OrderStatusCallRider)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err = getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").DeliveryFinish(param); err != nil {
|
|
globals.SugarLogger.Debugf("Swtich2SelfDeliver 出库失败可能是BizSubOrderId 没填写 : %s", err.Error())
|
|
return err
|
|
}
|
|
return err
|
|
}
|
|
|
|
// Swtich2SelfDelivered 订单送达
|
|
func (c *PurchaseHandler) Swtich2SelfDelivered(order *model.GoodsOrder, userName string) (err error) {
|
|
return err
|
|
}
|
|
|
|
// SelfDeliverDelivering 自配送订单配送中
|
|
func (c *PurchaseHandler) SelfDeliverDelivering(order *model.GoodsOrder, userName string) (err error) {
|
|
api := getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "")
|
|
// 出库
|
|
param, err := orderStatusChangeNotice(order, tao_vegetable.OrderStatusCallRider)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err = api.DeliveryFinish(param); err == nil {
|
|
paramDelivery, err := orderStatusChangeNotice(order, tao_vegetable.OrderStatusDelivery)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// 开始配送
|
|
return api.DeliveryFinish(paramDelivery)
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
// SelfDeliverDelivered 自配送订单送达
|
|
func (c *PurchaseHandler) SelfDeliverDelivered(order *model.GoodsOrder, userName string) (err error) {
|
|
param := OrderStatusChangeDelivery(order, tao_vegetable.OrderStatusDeliveryOver)
|
|
return getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").DeliveryFinish(param)
|
|
}
|
|
|
|
func (c *PurchaseHandler) GetOrderRealMobile(ctx *jxcontext.Context, order *model.GoodsOrder) (mobile string, err error) {
|
|
err = errors.New("淘菜菜外卖还未实现GetOrderRealMobile")
|
|
return mobile, err
|
|
}
|
|
|
|
func (c *PurchaseHandler) AgreeOrRefuseCancel(ctx *jxcontext.Context, order *model.GoodsOrder, isAgree bool, reason string) (err error) {
|
|
if isAgree {
|
|
param := &request3156.AlibabaTclsAelophyRefundAgreeRequest{
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
OutOrderId: utils.String2Pointer(order.VendorOrderID),
|
|
OrderFrom: utils.Int64ToPointer(utils.Str2Int64(tao_vegetable.TaoVegetableChannelCode)),
|
|
}
|
|
param.AuditMemo = utils.String2Pointer(fmt.Sprintf("商户同意退款"))
|
|
if reason != "" {
|
|
param.AuditMemo = utils.String2Pointer(*param.AuditMemo + fmt.Sprintf(",%s", reason))
|
|
}
|
|
err = getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").AgreeUserCancel(param)
|
|
} else {
|
|
afsOrder, err := partner.CurOrderManager.LoadAfsOrder(order.VendorOrderID, order.VendorID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
param := &request3156.AlibabaTclsAelophyRefundDisagreeRequest{
|
|
RefundId: utils.String2Pointer(afsOrder.AfsOrderID),
|
|
RejectReason: utils.String2Pointer(reason),
|
|
OrderFrom: utils.Int64ToPointer(utils.Str2Int64(tao_vegetable.TaoVegetableChannelCode)),
|
|
}
|
|
err = getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "").DisAgreeUserCancel(param)
|
|
if err != nil {
|
|
afsOrder.Status = model.AfsOrderStatusFailed
|
|
afsOrder.VendorStatus = "老板拒绝"
|
|
afsOrder.ReasonDesc += reason + ","
|
|
dao.UpdateEntity(dao.GetDB(), afsOrder, "Status", "ReasonDesc", "VendorStatus")
|
|
}
|
|
}
|
|
return err
|
|
}
|
|
|
|
// CancelOrder 商户取消订单(取消订单全部商品)
|
|
func (c *PurchaseHandler) CancelOrder(ctx *jxcontext.Context, order *model.GoodsOrder, reason string) (err error) {
|
|
api := getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "")
|
|
orderSkuList, err := getOrderCancelList(api, order)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
outSubOrderIds := make([]string, 0, len(*orderSkuList.OutSubOrders))
|
|
for _, v := range *orderSkuList.OutSubOrders {
|
|
if *v.CanReverse {
|
|
outSubOrderIds = append(outSubOrderIds, *v.OutSubOrderId)
|
|
}
|
|
}
|
|
|
|
param := &request3156.AlibabaTclsAelophyRefundCsapplyRequest{
|
|
RefundCsApplyDTO: &domain3156.AlibabaTclsAelophyRefundCsapplyRefundCsApplyDto{
|
|
ReasonId: utils.Int64ToPointer(1111),
|
|
OutOrderId: utils.String2Pointer(order.VendorOrderID),
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
RequestId: utils.String2Pointer(fmt.Sprintf("%s%d", order.VendorStoreID, time.Now().UnixNano())),
|
|
OutSubOrderIds: &outSubOrderIds,
|
|
Memo: utils.String2Pointer(reason),
|
|
},
|
|
}
|
|
|
|
if err = api.PartialRefund(param); err == nil {
|
|
c.postFakeMsg(order.VendorOrderID, tao_vegetable.OrderStatusMerchantCancel, tao_vegetable.OrderStatusMerchantCancel)
|
|
}
|
|
|
|
// 发送取消状态,商户取消
|
|
err = api.DeliveryFinish(&request591.AlibabaAelophyOrderWorkCallbackRequest{
|
|
WorkCallbackRequest: &domain591.AlibabaAelophyOrderWorkCallbackWorkCallbackRequest{
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
BizOrderId: utils.Int64ToPointer(utils.Str2Int64(order.VendorOrderID)),
|
|
Status: utils.String2Pointer(tao_vegetable.OrderStatusMerchantCancel),
|
|
},
|
|
})
|
|
|
|
return err
|
|
}
|
|
|
|
// AdjustOrder 商户发起部分退款(取消订单部分商品)
|
|
func (c *PurchaseHandler) AdjustOrder(ctx *jxcontext.Context, order *model.GoodsOrder, removedSkuList []*model.OrderSku, reason string) (err error) {
|
|
api := getAPI(order.VendorOrgCode, jxutils.GetSaleStoreIDFromOrder(order), "")
|
|
// 美团外卖必须要确认订单后才能调整单
|
|
if order.Status < model.OrderStatusFinishedPickup {
|
|
if err = c.PickupGoods(order, false, ctx.GetUserName()); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// 获取订单的子订单id
|
|
requestParam := &request591.AlibabaAelophyOrderGetRequest{OrderGetRequest: &domain591.AlibabaAelophyOrderGetOrderGetRequest{
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
BizOrderId: utils.Int64ToPointer(utils.Str2Int64(order.VendorOrderID)),
|
|
}}
|
|
orderDetail, err := api.QueryOrderDetail(requestParam)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
skuMap := make(map[string]string, 0)
|
|
for _, v := range *orderDetail.SubOrderResponseList {
|
|
skuMap[*v.SkuCode] = *v.OutSubOrderId
|
|
}
|
|
|
|
// 获取需要退货商品的子订单id
|
|
refundSkuOrderID := make([]string, 0, len(removedSkuList))
|
|
for _, v := range removedSkuList {
|
|
if orderId, ok := skuMap[v.VendorSkuID]; ok {
|
|
refundSkuOrderID = append(refundSkuOrderID, orderId)
|
|
}
|
|
}
|
|
|
|
// 商家申请退货
|
|
param := &request3156.AlibabaTclsAelophyRefundCsapplyRequest{
|
|
RefundCsApplyDTO: &domain3156.AlibabaTclsAelophyRefundCsapplyRefundCsApplyDto{
|
|
ReasonId: utils.Int64ToPointer(1111),
|
|
OutOrderId: utils.String2Pointer(order.VendorOrderID),
|
|
StoreId: utils.String2Pointer(order.VendorStoreID),
|
|
RequestId: utils.String2Pointer(fmt.Sprintf("%s%d", order.VendorStoreID, time.Now().UnixNano())),
|
|
OutSubOrderIds: &refundSkuOrderID,
|
|
Memo: utils.String2Pointer(reason),
|
|
},
|
|
}
|
|
|
|
return api.PartialRefund(param)
|
|
}
|
|
|
|
// ListOrders 获取门店订单列表(补全遗漏订单)
|
|
func (c *PurchaseHandler) ListOrders(ctx *jxcontext.Context, vendorOrgCode string, parentTask tasksch.ITask, queryDate time.Time, vendorStoreID string) (vendorOrderIDs []string, err error) {
|
|
if utils.IsTimeZero(queryDate) {
|
|
return nil, fmt.Errorf("queryDate必须指定")
|
|
}
|
|
queryDate = utils.Time2Date(queryDate)
|
|
|
|
var vendorStoreIDs []string
|
|
var api = getAPI(vendorOrgCode, 0, vendorStoreID)
|
|
if vendorStoreID == "" {
|
|
vendorStoreIDs, err = c.GetAllStoresVendorID(ctx, vendorOrgCode)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
vendorStoreIDs = []string{vendorStoreID}
|
|
}
|
|
task := tasksch.NewParallelTask("tao ListOrders", nil, ctx,
|
|
func(task *tasksch.ParallelTask, batchItemList []interface{}, params ...interface{}) (retVal interface{}, err error) {
|
|
vendorStoreID := batchItemList[0].(string)
|
|
var orderIDs []string
|
|
// 获取当前门店当天的订单
|
|
if vendorOderID, err2 := api.GetTransactionOrderListByTime(vendorStoreID, queryDate); err2 == nil {
|
|
orderIDs = vendorOderID
|
|
}
|
|
retVal = orderIDs
|
|
return retVal, nil
|
|
}, vendorStoreIDs)
|
|
tasksch.HandleTask(task, parentTask, true).Run()
|
|
orderList, err := task.GetResult(0)
|
|
if err == nil && len(orderList) > 0 {
|
|
vendorOrderIDs = make([]string, len(orderList))
|
|
for k, v := range orderList {
|
|
vendorOrderIDs[k] = v.(string)
|
|
}
|
|
}
|
|
return vendorOrderIDs, err
|
|
}
|
|
|
|
// GetWaybillTip 自配送暂无小费
|
|
func (c *PurchaseHandler) GetWaybillTip(ctx *jxcontext.Context, vendorOrgCode, vendorStoreID, vendorOrderID, vendorWaybillID, vendorWaybillID2 string) (tipFee int64, err error) {
|
|
return tipFee, err
|
|
}
|
|
|
|
// UpdateWaybillTip 暂无小费
|
|
func (c *PurchaseHandler) UpdateWaybillTip(ctx *jxcontext.Context, vendorOrgCode, vendorStoreID, vendorOrderID, vendorWaybillID, vendorWaybillID2, cityCode string, tipFee int64) (err error) {
|
|
return err
|
|
}
|
|
|
|
func (c *PurchaseHandler) GetSelfTakeCode(ctx *jxcontext.Context, order *model.GoodsOrder) (selfTakeCode string, err error) {
|
|
return selfTakeCode, err
|
|
}
|
|
|
|
func (c *PurchaseHandler) ConfirmSelfTake(ctx *jxcontext.Context, order *model.GoodsOrder, selfTakeCode string) (err error) {
|
|
return err
|
|
}
|
|
|
|
func (c *PurchaseHandler) ComplaintRider(vendorOrderId string, resonID int, resonContent string) (err error) {
|
|
return err
|
|
}
|
|
|
|
// GetCancelDeliveryReason 转自配送时取消非专送混合送门店取消理由
|
|
func (c *PurchaseHandler) GetCancelDeliveryReason(order *model.GoodsOrder) (string, error) {
|
|
return "", nil
|
|
}
|
|
|
|
// 取消美团外卖理由转使用三方配送
|
|
func (c *PurchaseHandler) CancelLogisticsByWmOrderId(order *model.GoodsOrder, reasonCode, detailContent, appPoiCode, orderId string) error {
|
|
return nil
|
|
}
|
|
|
|
// 获取订单配送状态
|
|
func (c *PurchaseHandler) OrderLogisticsStatus(orderId int64) (int64, error) {
|
|
return 0, nil
|
|
}
|
|
|
|
// GetOrderSettleAccounts 获取订单结算信息(t+1)当前订单获取不到结算
|
|
func (c *PurchaseHandler) GetOrderSettleAccounts(order *model.GoodsOrder) (int64, error) {
|
|
return 0, nil
|
|
}
|
|
|
|
// GetOrderTotalShopMoney 获取门店结算信息
|
|
func GetOrderTotalShopMoney(appOrgCode string, vendorStoreID string, start, end time.Time) (map[string]string, error) {
|
|
if appOrgCode == "" || vendorStoreID == "" {
|
|
return nil, errors.New("appKey 不能为空且平台门店ID不能为空")
|
|
}
|
|
if start.IsZero() || end.IsZero() {
|
|
return nil, errors.New("开始时间和结束时间不能为空")
|
|
}
|
|
|
|
settlement := make(map[string]string, 0)
|
|
|
|
api := getAPI(appOrgCode, 0, vendorStoreID)
|
|
startBillDate := util.LocalTime(start)
|
|
endBillDate := util.LocalTime(end)
|
|
pageSize := 200
|
|
pageIndex := 1
|
|
|
|
param := &request591.AlibabaWdkBillListRequest{
|
|
TxdBillListGetRequest: &domain591.AlibabaWdkBillListTxdBillListGetRequest{
|
|
EndBillDate: &endBillDate,
|
|
StartBillDate: &startBillDate,
|
|
ShopCode: utils.String2Pointer(vendorStoreID),
|
|
PageSize: utils.Int64ToPointer(int64(pageSize)),
|
|
PageIndex: utils.Int64ToPointer(int64(pageIndex)),
|
|
},
|
|
}
|
|
|
|
var totalIndex int64 = 0
|
|
result, _ := api.QueryBillList(param)
|
|
for _, v := range *result.TxdBillDetailBOS {
|
|
settlement[*v.BizOrderId] = *v.ReceivableAmount
|
|
}
|
|
|
|
if *result.Total > int64(pageSize) {
|
|
totalIndex = *result.Total / int64(pageSize)
|
|
if *result.Total%int64(pageSize) != model.NO {
|
|
totalIndex += 1
|
|
}
|
|
|
|
for i := 2; i <= int(totalIndex); i++ {
|
|
param.TxdBillListGetRequest.PageIndex = utils.Int64ToPointer(int64(i))
|
|
result2, _ := api.QueryBillList(param)
|
|
for _, v := range *result2.TxdBillDetailBOS {
|
|
settlement[*v.BizOrderId] = *v.ReceivableAmount
|
|
}
|
|
}
|
|
}
|
|
|
|
return settlement, nil
|
|
}
|