门店评分并发修改

This commit is contained in:
Rosy-zhudan
2019-09-11 14:49:59 +08:00
parent 3784696334
commit bfaea1b998

View File

@@ -6,6 +6,7 @@ import (
"sync" "sync"
"time" "time"
"git.rosy.net.cn/baseapi"
"git.rosy.net.cn/baseapi/utils" "git.rosy.net.cn/baseapi/utils"
"git.rosy.net.cn/jx-callback/business/jxstore/cms" "git.rosy.net.cn/jx-callback/business/jxstore/cms"
"git.rosy.net.cn/jx-callback/business/jxutils" "git.rosy.net.cn/jx-callback/business/jxutils"
@@ -18,6 +19,7 @@ import (
const ( const (
EnableScheduleScoreStore = !true EnableScheduleScoreStore = !true
ParallelCount = 5
GoldMedalScore = 90 GoldMedalScore = 90
SilverMedalScore = 80 SilverMedalScore = 80
@@ -38,7 +40,8 @@ const (
AbsentGoodsOrderNormalRatio = 1.0 //百分比 AbsentGoodsOrderNormalRatio = 1.0 //百分比
StoreRangeGoodRadius = 2.0 //千米 StoreRangeGoodRadius = 2.0 //千米
StoreRangeBadRadius = 1.0 //千米 StoreRangeBadRadius = 1.0 //千米
SaleSkuPriceRatio = 90 //百分比 SaleSkuPriceRatio = 90 //千米
SaleSkuCheckRange = 5.0 //千米
) )
var ( var (
@@ -115,7 +118,7 @@ func (s *StoreScoreDataWrapper) SetData(storeID int, valueName string, value int
} }
//得到所有门店的商品 //得到所有门店的商品
func GetAllStoreSkus(ctx *jxcontext.Context, storeList []*cms.StoreExt) { func GetAllStoreSkus(ctx *jxcontext.Context, parentTask tasksch.ITask, storeList []*cms.StoreExt) {
allStoreSkusWrapper.InitData() allStoreSkusWrapper.InitData()
taskFunc := func(task *tasksch.ParallelTask, batchItemList []interface{}, params ...interface{}) (retVal interface{}, err error) { taskFunc := func(task *tasksch.ParallelTask, batchItemList []interface{}, params ...interface{}) (retVal interface{}, err error) {
storeInfo := batchItemList[0].(*cms.StoreExt) storeInfo := batchItemList[0].(*cms.StoreExt)
@@ -133,8 +136,8 @@ func GetAllStoreSkus(ctx *jxcontext.Context, storeList []*cms.StoreExt) {
allStoreSkusWrapper.SetData(storeID, jxSkuMapData) allStoreSkusWrapper.SetData(storeID, jxSkuMapData)
return retVal, err return retVal, err
} }
taskParallel := tasksch.NewParallelTask("GetAllStoreSkus", nil, ctx, taskFunc, storeList) taskParallel := tasksch.NewParallelTask("得到所有门店商品", nil, ctx, taskFunc, storeList)
tasksch.HandleTask(taskParallel, nil, true).Run() tasksch.HandleTask(taskParallel, parentTask, true).Run()
taskParallel.GetResult(0) taskParallel.GetResult(0)
} }
@@ -153,8 +156,7 @@ func GetOpenTime(opTimeList []int16) (opTime float64) {
} }
//营业时间12小时及以上满分总分10分每少一个小时扣1分 //营业时间12小时及以上满分总分10分每少一个小时扣1分
func ScoreStoreOpenTime(storeList []*cms.StoreExt) { func ScoreStoreOpenTime(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
storeID := storeInfo.ID storeID := storeInfo.ID
storeStatus := storeInfo.Status storeStatus := storeInfo.Status
isStoreOpen := storeStatus == model.StoreStatusOpened isStoreOpen := storeStatus == model.StoreStatusOpened
@@ -178,11 +180,9 @@ func ScoreStoreOpenTime(storeList []*cms.StoreExt) {
} }
} }
} }
}
//可售商品数量大于1000总分10分按比例扣 //可售商品数量大于1000总分10分按比例扣
func ScoreSaleSkuCount(ctx *jxcontext.Context, storeList []*cms.StoreExt) { func ScoreSaleSkuCount(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
storeID := storeInfo.ID storeID := storeInfo.ID
skusMapData := allStoreSkusWrapper.GetData(storeID) skusMapData := allStoreSkusWrapper.GetData(storeID)
if len(skusMapData) > 0 { if len(skusMapData) > 0 {
@@ -203,11 +203,9 @@ func ScoreSaleSkuCount(ctx *jxcontext.Context, storeList []*cms.StoreExt) {
storeScoreDataWrapper.SetData(storeID, model.FieldSaleSkuCount, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldSaleSkuCount, finalScore)
} }
} }
}
//平均捡货时间小于等于拣货截止时间为10分满分每超出1分钟减1分 //平均捡货时间小于等于拣货截止时间为10分满分每超出1分钟减1分
func ScoreAveragePickupTime(storeList []*cms.StoreExt) { func ScoreAveragePickupTime(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
storeID := storeInfo.ID storeID := storeInfo.ID
db := dao.GetDB() db := dao.GetDB()
orderList, err := dao.GetDailyFinishOrderList(db, storeID) orderList, err := dao.GetDailyFinishOrderList(db, storeID)
@@ -232,11 +230,9 @@ func ScoreAveragePickupTime(storeList []*cms.StoreExt) {
storeScoreDataWrapper.SetData(storeID, model.FieldAveragePickupTime, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldAveragePickupTime, finalScore)
} }
} }
}
//差评订单和完成订单比例小于0.2%,得满分10分每增加0.1%减1分 //差评订单和完成订单比例小于0.2%,得满分10分每增加0.1%减1分
func ScoreBadCommentOrder(storeList []*cms.StoreExt) { func ScoreBadCommentOrder(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
storeID := storeInfo.ID storeID := storeInfo.ID
db := dao.GetDB() db := dao.GetDB()
badCommentOrderCount, _ := dao.GetDailyBadCommentOrderCount(db, storeID) badCommentOrderCount, _ := dao.GetDailyBadCommentOrderCount(db, storeID)
@@ -256,11 +252,9 @@ func ScoreBadCommentOrder(storeList []*cms.StoreExt) {
storeScoreDataWrapper.SetData(storeID, model.FieldBadCommentOrder, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldBadCommentOrder, finalScore)
} }
} }
}
//未完成订单和完成订单比小于1%得满分10分比例每增加5%分数减1 //未完成订单和完成订单比小于1%得满分10分比例每增加5%分数减1
func ScoreUnfinishOrder(storeList []*cms.StoreExt) { func ScoreUnfinishOrder(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
storeID := storeInfo.ID storeID := storeInfo.ID
db := dao.GetDB() db := dao.GetDB()
unFinishOrderCount, _ := dao.GetDailyUnFinishOrderCount(db, storeID) unFinishOrderCount, _ := dao.GetDailyUnFinishOrderCount(db, storeID)
@@ -280,11 +274,9 @@ func ScoreUnfinishOrder(storeList []*cms.StoreExt) {
storeScoreDataWrapper.SetData(storeID, model.FieldUnfinishOrder, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldUnfinishOrder, finalScore)
} }
} }
}
//缺货订单和完成订单比小于1%得10分比例每增加0.1%减1分 //缺货订单和完成订单比小于1%得10分比例每增加0.1%减1分
func ScoreAbsentGoodsOrder(storeList []*cms.StoreExt) { func ScoreAbsentGoodsOrder(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
storeID := storeInfo.ID storeID := storeInfo.ID
db := dao.GetDB() db := dao.GetDB()
absentGoodsOrderCount, _ := dao.GetDailyAbsentGoodsOrderCount(db, storeID) absentGoodsOrderCount, _ := dao.GetDailyAbsentGoodsOrderCount(db, storeID)
@@ -304,11 +296,9 @@ func ScoreAbsentGoodsOrder(storeList []*cms.StoreExt) {
storeScoreDataWrapper.SetData(storeID, model.FieldAbsentGoodsOrder, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldAbsentGoodsOrder, finalScore)
} }
} }
}
//促销品数量20个以上为满分10分每少2个扣1分 //促销品数量20个以上为满分10分每少2个扣1分
func ScorePromotionSku(storeList []*cms.StoreExt) { func ScorePromotionSku(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
storeID := storeInfo.ID storeID := storeInfo.ID
db := dao.GetDB() db := dao.GetDB()
beginTime := time.Now() beginTime := time.Now()
@@ -335,12 +325,10 @@ func ScorePromotionSku(storeList []*cms.StoreExt) {
storeScoreDataWrapper.SetData(storeID, model.FieldPromotionSku, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldPromotionSku, finalScore)
} }
} }
}
//经营全平台满分10分每少一个平台扣2分(一个平台没有得0分) //经营全平台满分10分每少一个平台扣2分(一个平台没有得0分)
func ScoreFullVendor(storeList []*cms.StoreExt) { func ScoreFullVendor(storeInfo *cms.StoreExt) {
fullVendorCount := len(fullVendorList) fullVendorCount := len(fullVendorList)
for _, storeInfo := range storeList {
finalScore := 0 finalScore := 0
storeID := storeInfo.ID storeID := storeInfo.ID
storeStatus := storeInfo.Status storeStatus := storeInfo.Status
@@ -354,7 +342,6 @@ func ScoreFullVendor(storeList []*cms.StoreExt) {
count++ count++
} }
} }
if count > 0 { if count > 0 {
if count == fullVendorCount { if count == fullVendorCount {
finalScore = ItemTotalScore finalScore = ItemTotalScore
@@ -365,11 +352,9 @@ func ScoreFullVendor(storeList []*cms.StoreExt) {
} }
storeScoreDataWrapper.SetData(storeID, model.FieldFullVendor, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldFullVendor, finalScore)
} }
}
//经营范围面积大于半径2km的圆得满分10分低于1km得分0 //经营范围面积大于半径2km的圆得满分10分低于1km得分0
func ScoreStoreRange(storeList []*cms.StoreExt) { func ScoreStoreRange(storeInfo *cms.StoreExt) {
for _, storeInfo := range storeList {
finalScore := 0 finalScore := 0
storeID := storeInfo.ID storeID := storeInfo.ID
if storeInfo.DeliveryRangeType == model.DeliveryRangeTypePolygon { if storeInfo.DeliveryRangeType == model.DeliveryRangeTypePolygon {
@@ -387,7 +372,7 @@ func ScoreStoreRange(storeList []*cms.StoreExt) {
finalScore = ItemTotalScore - int(math.Round(diff*ratio)) finalScore = ItemTotalScore - int(math.Round(diff*ratio))
} }
} else if storeInfo.DeliveryRangeType == model.DeliveryRangeTypeRadius { } else if storeInfo.DeliveryRangeType == model.DeliveryRangeTypeRadius {
deliveryRadius := utils.Str2Float64(storeInfo.DeliveryRange) / 1000 deliveryRadius := utils.Str2Float64WithDefault(storeInfo.DeliveryRange, 0) / 1000
if deliveryRadius >= StoreRangeGoodRadius { if deliveryRadius >= StoreRangeGoodRadius {
finalScore = ItemTotalScore finalScore = ItemTotalScore
} else if deliveryRadius <= StoreRangeBadRadius { } else if deliveryRadius <= StoreRangeBadRadius {
@@ -400,16 +385,15 @@ func ScoreStoreRange(storeList []*cms.StoreExt) {
} }
storeScoreDataWrapper.SetData(storeID, model.FieldStoreRange, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldStoreRange, finalScore)
} }
}
//得到距离某个门店多少KM内的所有门店信息 //得到距离某个门店多少KM内的所有门店信息
func GetRangeStoreList(storeID int, lng, lat, checkDist float64, storeList []*cms.StoreExt) (outStoreList []*cms.StoreExt) { func GetRangeStoreList(storeID int, lng, lat, checkRange float64, storeList []*cms.StoreExt) (outStoreList []*cms.StoreExt) {
for _, storeInfo := range storeList { for _, storeInfo := range storeList {
if storeInfo.ID == storeID { if storeInfo.ID == storeID {
outStoreList = append(outStoreList, storeInfo) outStoreList = append(outStoreList, storeInfo)
} else { } else {
distance := jxutils.EarthDistance(lng, lat, storeInfo.FloatLng, storeInfo.FloatLat) distance := jxutils.EarthDistance(lng, lat, storeInfo.FloatLng, storeInfo.FloatLat)
if distance <= checkDist { if distance <= checkRange {
outStoreList = append(outStoreList, storeInfo) outStoreList = append(outStoreList, storeInfo)
} }
} }
@@ -454,13 +438,12 @@ func GetSkusCountLessEqualAvgPrice(storeID int, skusAveragePrice map[int]int) (c
} }
//可售商品价格在附近5km内门店比较价格低于等于平均值的商品数占90%以上满分10分比例每降低10%减1分100%超标得0分 //可售商品价格在附近5km内门店比较价格低于等于平均值的商品数占90%以上满分10分比例每降低10%减1分100%超标得0分
func ScoreSaleSkuPrice(ctx *jxcontext.Context, storeList []*cms.StoreExt) { func ScoreSaleSkuPrice(storeInfo *cms.StoreExt, storeList []*cms.StoreExt) {
for _, storeInfo := range storeList {
finalScore := 0 finalScore := 0
storeID := storeInfo.ID storeID := storeInfo.ID
totalCount := len(allStoreSkusWrapper.GetData(storeID)) totalCount := len(allStoreSkusWrapper.GetData(storeID))
if totalCount > 0 { if totalCount > 0 {
rangeStoreList := GetRangeStoreList(storeID, storeInfo.FloatLng, storeInfo.FloatLat, 5, storeList) rangeStoreList := GetRangeStoreList(storeID, storeInfo.FloatLng, storeInfo.FloatLat, SaleSkuCheckRange, storeList)
skusAveragePrice := GetStoreSkusAveragePrice(rangeStoreList) skusAveragePrice := GetStoreSkusAveragePrice(rangeStoreList)
count := GetSkusCountLessEqualAvgPrice(storeID, skusAveragePrice) count := GetSkusCountLessEqualAvgPrice(storeID, skusAveragePrice)
if count > 0 { if count > 0 {
@@ -478,7 +461,6 @@ func ScoreSaleSkuPrice(ctx *jxcontext.Context, storeList []*cms.StoreExt) {
storeScoreDataWrapper.SetData(storeID, model.FieldSaleSkuPrice, finalScore) storeScoreDataWrapper.SetData(storeID, model.FieldSaleSkuPrice, finalScore)
} }
} }
}
func GetFilterStoreListEx(storeList []*cms.StoreExt, storeIDMap map[int]int) (outStoreList []*cms.StoreExt) { func GetFilterStoreListEx(storeList []*cms.StoreExt, storeIDMap map[int]int) (outStoreList []*cms.StoreExt) {
for _, storeInfo := range storeList { for _, storeInfo := range storeList {
@@ -506,24 +488,55 @@ func GetFilterStoreListEx(storeList []*cms.StoreExt, storeIDMap map[int]int) (ou
} }
func ScoreStore(ctx *jxcontext.Context, storeIDList []int) (retVal interface{}, err error) { func ScoreStore(ctx *jxcontext.Context, storeIDList []int) (retVal interface{}, err error) {
var storeList []*cms.StoreExt
taskCount := 4
taskSeqFunc := func(task *tasksch.SeqTask, step int, params ...interface{}) (result interface{}, err error) {
switch step {
case 0:
baseapi.SugarLogger.Debugf("ScoreStore step0 begin")
storeScoreDataWrapper.InitData() storeScoreDataWrapper.InitData()
storeList, err := GetStoreList(ctx) storeList, err = GetStoreList(ctx)
storeIDMap := jxutils.IntList2Map(storeIDList) storeIDMap := jxutils.IntList2Map(storeIDList)
storeList = GetFilterStoreListEx(storeList, storeIDMap) storeList = GetFilterStoreListEx(storeList, storeIDMap)
baseapi.SugarLogger.Debugf("ScoreStore step0 end")
GetAllStoreSkus(ctx, storeList) case 1:
ScoreStoreOpenTime(storeList) baseapi.SugarLogger.Debugf("ScoreStore step1 begin")
ScoreSaleSkuCount(ctx, storeList) GetAllStoreSkus(ctx, task, storeList)
ScoreAveragePickupTime(storeList) baseapi.SugarLogger.Debugf("ScoreStore step1 end")
ScoreBadCommentOrder(storeList) case 2:
ScoreUnfinishOrder(storeList) baseapi.SugarLogger.Debugf("ScoreStore step2 begin")
ScoreAbsentGoodsOrder(storeList) taskFunc := func(task *tasksch.ParallelTask, batchItemList []interface{}, params ...interface{}) (retVal interface{}, err error) {
ScorePromotionSku(storeList) storeInfo := batchItemList[0].(*cms.StoreExt)
ScoreFullVendor(storeList) ScoreStoreOpenTime(storeInfo)
ScoreStoreRange(storeList) ScoreSaleSkuCount(storeInfo)
ScoreSaleSkuPrice(ctx, storeList) ScoreAveragePickupTime(storeInfo)
ScoreBadCommentOrder(storeInfo)
ScoreUnfinishOrder(storeInfo)
ScoreAbsentGoodsOrder(storeInfo)
ScorePromotionSku(storeInfo)
ScoreFullVendor(storeInfo)
ScoreStoreRange(storeInfo)
ScoreSaleSkuPrice(storeInfo, storeList)
return retVal, err
}
taskParallel := tasksch.NewParallelTask("计算门店得分", tasksch.NewParallelConfig().SetParallelCount(ParallelCount), ctx, taskFunc, storeList)
tasksch.HandleTask(taskParallel, task, true).Run()
taskParallel.GetResult(0)
_, err = taskParallel.GetResult(0)
if err != nil {
baseapi.SugarLogger.Debugf("ScoreStore taskParallel error:%v", err)
}
baseapi.SugarLogger.Debugf("ScoreStore step2 end")
case 3:
baseapi.SugarLogger.Debugf("ScoreStore step3 begin")
InsertStoreScore() InsertStoreScore()
allStoreSkusWrapper.ClearData() allStoreSkusWrapper.ClearData()
baseapi.SugarLogger.Debugf("ScoreStore step3 end")
}
return result, err
}
taskSeq := tasksch.NewSeqTask("门店评分-序列任务", ctx, taskSeqFunc, taskCount)
tasksch.HandleTask(taskSeq, nil, true).Run()
return retVal, err return retVal, err
} }