426 lines
15 KiB
Go
426 lines
15 KiB
Go
package ebai
|
||
|
||
import (
|
||
"fmt"
|
||
"strings"
|
||
|
||
"git.rosy.net.cn/baseapi/platformapi/autonavi"
|
||
"git.rosy.net.cn/baseapi/platformapi/ebaiapi"
|
||
"git.rosy.net.cn/baseapi/utils"
|
||
"git.rosy.net.cn/jx-callback/business/jxcallback/scheduler"
|
||
"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"
|
||
"git.rosy.net.cn/jx-callback/globals/api"
|
||
)
|
||
|
||
const (
|
||
VendorStorePrefix = "好菜鲜生"
|
||
)
|
||
|
||
type tEbaiStoreInfo struct {
|
||
model.Store
|
||
VendorStoreID string `orm:"column(vendor_store_id)"`
|
||
RealLastOperator string
|
||
EbaiStoreStatus int
|
||
SyncStatus int
|
||
|
||
ProvinceID int `orm:"column(province_id)"`
|
||
CityID int `orm:"column(city_id)"`
|
||
DistrictID int `orm:"column(district_id)"`
|
||
}
|
||
|
||
func (p *PurchaseHandler) CreateStore(db *dao.DaoDB, storeID int, userName string) (vendorStoreID string, err error) {
|
||
var store tEbaiStoreInfo
|
||
sql := `
|
||
SELECT t1.*, t2.status ebai_store_status, t2.vendor_store_id,
|
||
IF(t1.updated_at > t2.updated_at, t1.last_operator, t2.last_operator) real_last_operator, t2.sync_status,
|
||
province.ebai_code province_id, city.ebai_code city_id, district.ebai_code district_id
|
||
FROM store t1
|
||
LEFT JOIN store_map t2 ON t1.id = t2.store_id AND t2.vendor_id = ? AND t2.deleted_at = ?
|
||
JOIN place district ON t1.district_code = district.code
|
||
JOIN place city ON t1.city_code = city.code
|
||
JOIN place province ON city.parent_code = province.code
|
||
WHERE t1.id = ? AND t2.id IS NULL;
|
||
`
|
||
if err = dao.GetRow(db, &store, sql, model.VendorIDEBAI, utils.DefaultTimeValue, storeID); err == nil {
|
||
params := genStoreMapFromStore(&store)
|
||
params["shop_id"] = store.ID
|
||
params["business_form_id"] = "179"
|
||
params["service_phone"] = store.Tel1
|
||
params["invoice_support"] = 2
|
||
params["package_box_price"] = 0
|
||
params["encrypt"] = ""
|
||
|
||
params["category1"] = ""
|
||
params["category2"] = ""
|
||
params["category3"] = ""
|
||
if globals.EnableEbaiStoreWrite {
|
||
intVendorStoreID, err2 := api.EbaiAPI.ShopCreate(params)
|
||
if err = err2; err == nil {
|
||
return utils.Int64ToStr(intVendorStoreID), err
|
||
}
|
||
} else {
|
||
return utils.Int64ToStr(jxutils.GenFakeID()), nil
|
||
}
|
||
}
|
||
return "", err
|
||
}
|
||
|
||
func (p *PurchaseHandler) ReadStore(vendorStoreID string) (*model.Store, error) {
|
||
baiduShopID := utils.Str2Int64(vendorStoreID)
|
||
result, err := api.EbaiAPI.ShopGet("", baiduShopID)
|
||
if err == nil {
|
||
// globals.SugarLogger.Debug(utils.Format4Output(result, false))
|
||
retVal := &model.Store{
|
||
Address: utils.Interface2String(result["address"]),
|
||
Tel1: utils.Interface2String(result["phone"]),
|
||
}
|
||
retVal.OriginalName = utils.Interface2String(result["name"])
|
||
_, retVal.Name = jxutils.SplitStoreName(retVal.OriginalName, partner.StoreNameSeparator, VendorStorePrefix)
|
||
retVal.DeliveryType = EbaiDeliveryType2Jx(utils.Interface2String(result["delivery_type"]))
|
||
|
||
if result["business_time"] != nil {
|
||
businessTime := result["business_time"].([]interface{})
|
||
if len(businessTime) > 0 {
|
||
btime := businessTime[0].(map[string]interface{})
|
||
retVal.OpenTime1 = jxutils.StrTime2JxOperationTime(utils.Interface2String(btime["start"].(string)), 700)
|
||
retVal.CloseTime1 = jxutils.StrTime2JxOperationTime(utils.Interface2String(btime["end"].(string)), 2000)
|
||
if len(businessTime) > 1 {
|
||
btime = businessTime[1].(map[string]interface{})
|
||
retVal.OpenTime2 = jxutils.StrTime2JxOperationTime(utils.Interface2String(btime["start"].(string)), 700)
|
||
retVal.CloseTime2 = jxutils.StrTime2JxOperationTime(utils.Interface2String(btime["end"].(string)), 2000)
|
||
}
|
||
}
|
||
}
|
||
|
||
if ebaiStatus, err2 := api.EbaiAPI.ShopBusStatusGet("", baiduShopID, ebaiapi.PlatformFlagElm); err2 == nil {
|
||
retVal.Status = EbaiBusStatus2JxStatus(ebaiStatus)
|
||
}
|
||
|
||
tel2 := utils.Interface2String(result["ivr_phone"])
|
||
if tel2 != "" && tel2 != retVal.Tel1 {
|
||
retVal.Tel2 = tel2
|
||
}
|
||
|
||
lng := utils.MustInterface2Float64(result["longitude"])
|
||
lat := utils.MustInterface2Float64(result["latitude"])
|
||
if utils.Interface2String(result["coord_type"]) == ebaiapi.CoordTypeBaidu {
|
||
var err2 error
|
||
if lng, lat, err2 = api.AutonaviAPI.CoordinateConvert(lng, lat, autonavi.CoordSysBaidu); err2 != nil {
|
||
return nil, err2
|
||
}
|
||
}
|
||
retVal.Lng = jxutils.StandardCoordinate2Int(lng)
|
||
retVal.Lat = jxutils.StandardCoordinate2Int(lat)
|
||
|
||
db := dao.GetDB()
|
||
if city, err2 := dao.GetPlaceByName(db, utils.Interface2String(result["city"]), model.PlaceLevelCity, 0); err2 == nil {
|
||
retVal.CityCode = city.Code
|
||
districtName := utils.Interface2String(result["county"])
|
||
if retVal.CityCode != 0 && districtName != "" {
|
||
if district, err2 := dao.GetPlaceByName(db, utils.Interface2String(result["county"]), model.PlaceLevelDistrict, city.Code); err2 == nil {
|
||
retVal.DistrictCode = district.Code
|
||
}
|
||
}
|
||
}
|
||
if retVal.DistrictCode == 0 {
|
||
retVal.DistrictCode = api.AutonaviAPI.GetCoordinateDistrictCode(lng, lat)
|
||
if retVal.CityCode == 0 {
|
||
if district, err := dao.GetPlaceByCode(db, retVal.DistrictCode); err == nil {
|
||
retVal.CityCode = district.ParentCode
|
||
}
|
||
}
|
||
}
|
||
retVal.ID = int(utils.Str2Int64WithDefault(utils.Interface2String(result["shop_id"]), 0))
|
||
retVal.DeliveryRangeType = model.DeliveryRangeTypePolygon
|
||
retVal.DeliveryRange = EbaiDeliveryRegion2Jx(result["delivery_region"])
|
||
|
||
return retVal, nil
|
||
}
|
||
return nil, err
|
||
}
|
||
|
||
func (p *PurchaseHandler) UpdateStore(db *dao.DaoDB, storeID int, userName string) (err error) {
|
||
globals.SugarLogger.Debugf("ebai UpdateStore storeID:%d, userName:%s", storeID, userName)
|
||
|
||
var stores []*tEbaiStoreInfo
|
||
sql := `
|
||
SELECT t1.*, t2.status ebai_store_status, t2.vendor_store_id,
|
||
IF(t1.updated_at > t2.updated_at, t1.last_operator, t2.last_operator) real_last_operator, t2.sync_status
|
||
FROM store t1
|
||
JOIN store_map t2 ON t1.id = t2.store_id AND t2.vendor_id = ? AND (t2.deleted_at = ?)
|
||
WHERE t1.id = ?
|
||
ORDER BY t2.updated_at
|
||
`
|
||
if err = dao.GetRows(db, &stores, sql, model.VendorIDEBAI, utils.DefaultTimeValue, storeID); err == nil {
|
||
for _, store := range stores {
|
||
// globals.SugarLogger.Debug(utils.Format4Output(params, false))
|
||
if globals.EnableEbaiStoreWrite {
|
||
shopID := 0
|
||
if store.SyncStatus&model.SyncFlagDeletedMask == 0 {
|
||
shopID = store.ID
|
||
}
|
||
store2, err2 := p.ReadStore(store.VendorStoreID)
|
||
// globals.SugarLogger.Debugf("ebai UpdateStore2 store2:%s, err2:%v", utils.Format4Output(store2, true), err2)
|
||
if err = err2; err == nil {
|
||
if store2.ID == store.ID {
|
||
shopID = -1
|
||
}
|
||
}
|
||
if err == nil {
|
||
if shopID > 0 {
|
||
err = api.EbaiAPI.ShopIDBatchUpdate([]string{store.VendorStoreID}, []string{utils.Int2Str(shopID)})
|
||
} else if shopID == 0 {
|
||
// todo remove out shop id
|
||
}
|
||
}
|
||
if err == nil {
|
||
mergeStatus := jxutils.MergeStoreStatus(store.Status, store.EbaiStoreStatus)
|
||
if store2.Status != mergeStatus {
|
||
if mergeStatus == model.StoreStatusOpened {
|
||
err = api.EbaiAPI.ShopOpen("", utils.Str2Int64(store.VendorStoreID))
|
||
} else if mergeStatus == model.StoreStatusClosed {
|
||
err = api.EbaiAPI.ShopClose("", utils.Str2Int64(store.VendorStoreID))
|
||
} else if mergeStatus == model.StoreStatusDisabled {
|
||
err = api.EbaiAPI.ShopOffline("", utils.Str2Int64(store.VendorStoreID))
|
||
}
|
||
if intErr, ok := err.(*utils.ErrorWithCode); ok && intErr.IntCode() == 201100 {
|
||
err = nil
|
||
}
|
||
}
|
||
if err != nil {
|
||
// todo
|
||
if intErr, ok := err.(*utils.ErrorWithCode); ok {
|
||
if intErr.IntCode() == 201101 {
|
||
err = nil
|
||
}
|
||
}
|
||
}
|
||
if err != nil {
|
||
return err
|
||
}
|
||
// todo 饿百 开店审核通过后不允许修改商户信息
|
||
// params := genStoreMapFromStore(store)
|
||
// if err = api.EbaiAPI.ShopUpdate(params); err == nil {
|
||
// }
|
||
}
|
||
}
|
||
}
|
||
}
|
||
return err
|
||
}
|
||
|
||
func (p *PurchaseHandler) RefreshAllStoresID(ctx *jxcontext.Context, parentTask tasksch.ITask, isAsync bool) (hint string, err error) {
|
||
globals.SugarLogger.Debugf("ebai RefreshAllStoresID")
|
||
const batchSize = 50
|
||
const stepCount = 3
|
||
var stores []*tEbaiStoreInfo
|
||
db := dao.GetDB()
|
||
rootTask := tasksch.NewSeqTask("ebai RefreshAllStoresID", ctx,
|
||
func(task *tasksch.SeqTask, step int, params ...interface{}) (result interface{}, err error) {
|
||
switch step {
|
||
case 0:
|
||
err = dao.GetRows(db, &stores, `
|
||
SELECT t1.*, t2.vendor_store_id
|
||
FROM store t1
|
||
JOIN store_map t2 ON t1.id = t2.store_id AND t2.deleted_at = ? AND t2.vendor_id = ?
|
||
WHERE t1.deleted_at = ?
|
||
`, utils.DefaultTimeValue, model.VendorIDEBAI, utils.DefaultTimeValue)
|
||
default:
|
||
taskName := "ebai RefreshAllStoresID update to custom id"
|
||
if step != stepCount-1 {
|
||
taskName = "ebai RefreshAllStoresID update to uuid"
|
||
}
|
||
task1 := tasksch.NewParallelTask(taskName, tasksch.NewParallelConfig().SetBatchSize(batchSize), ctx,
|
||
func(task *tasksch.ParallelTask, batchItemList []interface{}, params ...interface{}) (retVal interface{}, err error) {
|
||
baiduShopIDs := make([]string, len(batchItemList))
|
||
shopIDs := make([]string, len(batchItemList))
|
||
for k, v := range batchItemList {
|
||
store := v.(*tEbaiStoreInfo)
|
||
baiduShopIDs[k] = store.VendorStoreID
|
||
shopIDs[k] = utils.Int2Str(store.ID)
|
||
if step != stepCount-1 {
|
||
shopIDs[k] = utils.GetUUID()
|
||
}
|
||
}
|
||
if globals.EnableEbaiStoreWrite {
|
||
err = api.EbaiAPI.ShopIDBatchUpdate(baiduShopIDs, shopIDs)
|
||
}
|
||
return nil, err
|
||
}, stores)
|
||
task.AddChild(task1).Run()
|
||
_, err = task1.GetResult(0)
|
||
}
|
||
return nil, err
|
||
}, stepCount)
|
||
|
||
tasksch.HandleTask(rootTask, parentTask, false).Run()
|
||
if !isAsync {
|
||
_, err = rootTask.GetResult(0)
|
||
}
|
||
return rootTask.ID, err
|
||
}
|
||
|
||
// todo 此函数只考虑了在饿了么侧开店的情况
|
||
func EbaiDeliveryType2Jx(deliveryType string) int8 {
|
||
spIndex := strings.Index(deliveryType, "|")
|
||
elmDeliveryType := utils.Str2Int64(deliveryType[:spIndex])
|
||
switch elmDeliveryType {
|
||
case ebaiapi.DeliveryTypeElmFengNiaoZS,
|
||
ebaiapi.DeliveryTypeElmFengNiaoZSKA,
|
||
ebaiapi.DeliveryTypeElmFengNiaoKS,
|
||
ebaiapi.DeliveryTypeElmNewRetail,
|
||
ebaiapi.DeliveryTypeElmEPeiSong,
|
||
ebaiapi.DeliveryTypeElmFengNiaoHybrid,
|
||
ebaiapi.DeliveryTypeElmFengNiaoNiubee:
|
||
return scheduler.StoreDeliveryTypeByPlatform
|
||
case ebaiapi.DeliveryTypeElmXingHuoZBTrial,
|
||
ebaiapi.DeliveryTypeElmXingHuoZB,
|
||
ebaiapi.DeliveryTypeElmXingHuoZBKA:
|
||
return scheduler.StoreDeliveryTypeCrowdSourcing
|
||
case ebaiapi.DeliveryTypeElmNone,
|
||
ebaiapi.DeliveryTypeElmXingHuoTrial,
|
||
ebaiapi.DeliveryTypeElmXingHuo,
|
||
ebaiapi.DeliveryTypeElmEBase,
|
||
ebaiapi.DeliveryTypeElmXingHuoKA:
|
||
return scheduler.StoreDeliveryTypeByStore
|
||
default:
|
||
return scheduler.StoreDeliveryTypeCrowdSourcing
|
||
}
|
||
}
|
||
|
||
func EbaiDeliveryRegion2Jx(deliveryRegion interface{}) string {
|
||
realDeliveryRegion := deliveryRegion.([]interface{})
|
||
if len(realDeliveryRegion) > 0 {
|
||
region := deliveryRegion.([]interface{})[0].(map[string]interface{})["region"].([]interface{})[0].([]interface{})
|
||
coords := make([]string, len(region))
|
||
for k, v := range region {
|
||
mapV := v.(map[string]interface{})
|
||
coords[k] = fmt.Sprintf("%.6f,%.6f", utils.MustInterface2Float64(mapV["longitude"]), utils.MustInterface2Float64(mapV["latitude"]))
|
||
}
|
||
return strings.Join(coords, ";")
|
||
}
|
||
return ""
|
||
}
|
||
|
||
func JxDeliveryRegion2Ebai(store *model.Store) (deliveryRegion interface{}) {
|
||
rangeStr := strings.Trim(store.DeliveryRange, ";")
|
||
if store.DeliveryRangeType == model.DeliveryRangeTypeRadius {
|
||
if utils.Str2Int64(store.DeliveryRange) > 100 { // todo 如果小于100米,表示禁用,不更新
|
||
rangeStr = jxutils.GetPolygonFromCircleStr(jxutils.IntCoordinate2Standard(store.Lng), jxutils.IntCoordinate2Standard(store.Lat), utils.Str2Float64(store.DeliveryRange), 8)
|
||
} else {
|
||
rangeStr = ""
|
||
}
|
||
}
|
||
rangeStr = "" // todo 暂时禁止同步配送区域
|
||
if rangeStr != "" {
|
||
pointPairs := strings.Split(rangeStr, ";")
|
||
region := make([]map[string]interface{}, 0)
|
||
for _, v := range pointPairs {
|
||
pointPair := strings.Split(v, ",")
|
||
if len(pointPair) == 2 {
|
||
region = append(region, map[string]interface{}{
|
||
"longitude": utils.Str2Float64(pointPair[0]),
|
||
"latitude": utils.Str2Float64(pointPair[1]),
|
||
})
|
||
}
|
||
}
|
||
deliveryRegion = []interface{}{
|
||
map[string]interface{}{
|
||
"name": "主要配送区",
|
||
"delivery_fee": 600,
|
||
"delivery_time": "60",
|
||
"min_buy_free": "0",
|
||
"min_order_price": "0",
|
||
"region": []interface{}{
|
||
region,
|
||
},
|
||
},
|
||
}
|
||
}
|
||
return deliveryRegion
|
||
}
|
||
|
||
func JxBusinessTime2Ebai(store *model.Store) interface{} {
|
||
bTime := make([]map[string]interface{}, 1)
|
||
bTime[0] = map[string]interface{}{
|
||
"start": jxutils.JxOperationTime2StrTime(store.OpenTime1),
|
||
"end": jxutils.JxOperationTime2StrTime(store.CloseTime1),
|
||
}
|
||
if store.OpenTime2 != 0 {
|
||
bTime = append(bTime, map[string]interface{}{
|
||
"start": jxutils.JxOperationTime2StrTime(store.OpenTime2),
|
||
"end": jxutils.JxOperationTime2StrTime(store.CloseTime2),
|
||
})
|
||
}
|
||
return bTime
|
||
}
|
||
|
||
func genStoreMapFromStore(store *tEbaiStoreInfo) map[string]interface{} {
|
||
params := map[string]interface{}{
|
||
"phone": store.Tel1,
|
||
"business_time": JxBusinessTime2Ebai(&store.Store),
|
||
}
|
||
if store.VendorStoreID != "" {
|
||
params["baidu_shop_id"] = store.VendorStoreID
|
||
}
|
||
if store.Tel2 != "" {
|
||
params["ivr_phone"] = store.Tel2
|
||
}
|
||
if store.SyncStatus&(model.SyncFlagNewMask|model.SyncFlagStoreName) != 0 {
|
||
// params["name"] = jxutils.ComposeStoreName(store.Name, model.VendorIDEBAI)
|
||
}
|
||
if store.SyncStatus&(model.SyncFlagNewMask|model.SyncFlagStoreAddress) != 0 {
|
||
params["longitude"] = jxutils.IntCoordinate2Standard(store.Lng)
|
||
params["latitude"] = jxutils.IntCoordinate2Standard(store.Lat)
|
||
params["address"] = store.Address
|
||
params["coord_type"] = ebaiapi.CoordTypeAutonavi
|
||
if deliveryRegion := JxDeliveryRegion2Ebai(&store.Store); deliveryRegion != nil {
|
||
params["delivery_region"] = deliveryRegion
|
||
}
|
||
if store.ProvinceID != 0 {
|
||
params["province"] = store.ProvinceID
|
||
}
|
||
if store.CityID != 0 {
|
||
params["city"] = store.CityID
|
||
}
|
||
if store.DistrictID != 0 {
|
||
params["county"] = store.DistrictID
|
||
}
|
||
}
|
||
return params
|
||
}
|
||
|
||
func (p *PurchaseHandler) GetStoreStatus(ctx *jxcontext.Context, vendorStoreID string) (storeStatus int, err error) {
|
||
ebaiStatus, err := api.EbaiAPI.ShopBusStatusGet("", utils.Str2Int64(vendorStoreID), ebaiapi.PlatformFlagElm)
|
||
if err == nil {
|
||
storeStatus = EbaiBusStatus2JxStatus(ebaiStatus)
|
||
}
|
||
return storeStatus, err
|
||
}
|
||
|
||
func (c *PurchaseHandler) onShopMsgPush(msg *ebaiapi.CallbackMsg) (response *ebaiapi.CallbackResponse) {
|
||
var err error
|
||
vendorStoreID := utils.Interface2String(msg.Body["baidu_shop_id"])
|
||
storeStatus := model.StoreStatusOpened
|
||
switch utils.Interface2String(msg.Body["msg_type"]) {
|
||
case "online", "offline":
|
||
storeStatus, err = c.GetStoreStatus(jxcontext.AdminCtx, vendorStoreID)
|
||
case "shop_open", "shop_pause", "shop_close":
|
||
if int(utils.ForceInterface2Int64(msg.Body["business_ele"])) == 1 {
|
||
storeStatus = model.StoreStatusOpened
|
||
} else {
|
||
storeStatus = model.StoreStatusClosed
|
||
}
|
||
}
|
||
if err == nil {
|
||
err = partner.CurStoreManager.OnStoreStatusChanged(vendorStoreID, model.VendorIDEBAI, storeStatus)
|
||
}
|
||
return api.EbaiAPI.Err2CallbackResponse(msg.Cmd, err, nil)
|
||
}
|