saga事务问题

saga事务问题

请问一下这里已经扣减库存成功了,为什么还会有一直重复调用扣减库存的情况呢?

感觉上是有多个线程一直在调用,saga事务状态一直是submitted状态

https://img1.sycdn.imooc.com//climg/649aacc809cff70912280202.jpg


如图:这里已经执行成功扣减库存的sell方法,并进行了421和422的库存扣减,创建了stockselldetail信息

https://img1.sycdn.imooc.com//climg/649aabe2095f9afe14330675.jpg

这里仍然会有其他线程的调用记录:

https://img1.sycdn.imooc.com//climg/649aac23096840ab13870742.jpg

正在回答

登陆购买课程后可参与讨论,去登陆

3回答

库存扣减成功需要一个返回告诉这个调用逻辑成功了否则dtm会反复重试

  • YLDALONG 提问者 #1

    扣减库存的事务commit后不是这样返回吗?怎样返回告诉这个调用逻辑成功呢?


    https://img1.sycdn.imooc.com//climg/649ce188092e231c15390639.jpg


    https://img1.sycdn.imooc.com//climg/649ce1cf0998e48a14750731.jpg

    2023-06-29 09:43:51
  • bobby 回复 提问者 YLDALONG #2

    只要这里返回的err不是nil就代表成功,你在return的时候打印一下看看是否返回的err是nil,同时记得这个接口不能超时,dtm对每个请求是有超时的,所以你不要打断点或者执行时间过长

    2023-06-29 22:33:55
  • YLDALONG 提问者 回复 bobby #3

    排查出是grpc调用设置的超时时间太短了,谢谢老师的提醒!

    2023-06-30 10:20:59
提问者 YLDALONG 2023-06-27 17:42:20

调用扣减库存的逻辑:


r := gin.Default()
r.GET("start", func(c *gin.Context) {
   orderSn := shortuuid.New()
   req := &proto.SellInfo{
      GoodsInfo: []*proto.GoodsInvInfo{
         {
            GoodsId: 421,
            Num:     2,
         },
         {
            GoodsId: 422,
            Num:     2,
         },
      },
      OrderSn: orderSn,
   }
   dmtServer := "192.168.56.1:36790"
   qsBusi := "discovery://192.168.159.142:8500/mxshop-inventory-srv"
   //qsBusi := "192.168.56.1:8019"
   fmt.Println(orderSn)
   saga := dtmgrpc.NewSagaGrpc(dmtServer, orderSn).
      Add(qsBusi+"/Inventory/Sell", qsBusi+"/Inventory/Reback", req)
   err := saga.Submit()
   if err != nil {
      c.JSON(500, gin.H{"message": err.Error()})
   }
   c.JSON(200, gin.H{"message": "ok"})
})

r.Run(":8089")


  • 提问者 YLDALONG #1

    扣减库存的逻辑:

    func (is *inventoryService) Sell(ctx context.Context, ordersn string, details []do.GoodsDetail) error {
       log.Infof("订单%s扣减库存", ordersn)
       //解决了空悬挂的问题
       //先查询刚才插入的记录是否存在,如果存在则说明已经cancel就不能执行了

       rs := redsync.New(is.pool)
       //实际上批量扣减库存的时候, 我们经常会先按照商品的id排序,然后从小大小逐个扣减库存,这样可以减少锁的竞争
       //如果无序的话 那么就有可能订单a 扣减 1,3,4 订单B 扣减 3,2,1
       var detail = do.GoodsDetailList(details)
       sort.Sort(detail)

       txn := is.data.Begin()
       defer func() {
          if err := recover(); err != nil {
             _ = txn.Rollback()
             log.Error("事务进行中出现异常,回滚")
             return
          }
       }()

       sellDetail := do.StockSellDetailDO{
          OrderSn: ordersn,
          Status:  1,
          Detail:  detail,
       }

       for _, goodsInfo := range detail {
          mutex := rs.NewMutex(inventoryLockPrefix + ordersn)
          if err := mutex.Lock(); err != nil {
             log.Errorf("订单%s获取锁失败", ordersn)
          }
          defer mutex.Unlock()

          inv, err := is.data.Inventorys().Get(ctx, uint64(goodsInfo.Goods))
          if err != nil {
             log.Errorf("订单%s获取库存失败", ordersn)
             return err
          }

          //判断库存是否充足
          if inv.Stocks < goodsInfo.Num {
             txn.Rollback() //回滚
             log.Errorf("商品%d库存%d不足, 现有库存: %d", goodsInfo.Goods, goodsInfo.Num, inv.Stocks)
             return errors.WithCode(code.ErrInvNotEnough, "库存不足")
          }
          inv.Stocks -= goodsInfo.Num

          err = is.data.Inventorys().Reduce(ctx, txn, uint64(goodsInfo.Goods), int(goodsInfo.Num))
          if err != nil {
             txn.Rollback() //回滚
             log.Errorf("订单%s扣减库存失败", ordersn)
             return err
          }

       }

       err := is.data.Inventorys().CreateStockSellDetail(ctx, txn, &sellDetail)
       if err != nil {
          txn.Rollback() //回滚
          log.Errorf("订单%s创建扣减库存记录失败", ordersn)
          return err
       }

       txn.Commit()
       return nil
    }


    2023-06-27 17:44:10
提问者 YLDALONG 2023-06-27 17:41:48

扣减库存的逻辑:


func (is *inventoryService) Sell(ctx context.Context, ordersn string, details []do.GoodsDetail) error {
   log.Infof("订单%s扣减库存", ordersn)
   //解决了空悬挂的问题
   //先查询刚才插入的记录是否存在,如果存在则说明已经cancel就不能执行了

   rs := redsync.New(is.pool)
   //实际上批量扣减库存的时候, 我们经常会先按照商品的id排序,然后从小大小逐个扣减库存,这样可以减少锁的竞争
   //如果无序的话 那么就有可能订单a 扣减 1,3,4 订单B 扣减 3,2,1
   var detail = do.GoodsDetailList(details)
   sort.Sort(detail)

   txn := is.data.Begin()
   defer func() {
      if err := recover(); err != nil {
         _ = txn.Rollback()
         log.Error("事务进行中出现异常,回滚")
         return
      }
   }()

   sellDetail := do.StockSellDetailDO{
      OrderSn: ordersn,
      Status:  1,
      Detail:  detail,
   }

   for _, goodsInfo := range detail {
      mutex := rs.NewMutex(inventoryLockPrefix + ordersn)
      if err := mutex.Lock(); err != nil {
         log.Errorf("订单%s获取锁失败", ordersn)
      }
      defer mutex.Unlock()

      inv, err := is.data.Inventorys().Get(ctx, uint64(goodsInfo.Goods))
      if err != nil {
         log.Errorf("订单%s获取库存失败", ordersn)
         return err
      }

      //判断库存是否充足
      if inv.Stocks < goodsInfo.Num {
         txn.Rollback() //回滚
         log.Errorf("商品%d库存%d不足, 现有库存: %d", goodsInfo.Goods, goodsInfo.Num, inv.Stocks)
         return errors.WithCode(code.ErrInvNotEnough, "库存不足")
      }
      inv.Stocks -= goodsInfo.Num

      err = is.data.Inventorys().Reduce(ctx, txn, uint64(goodsInfo.Goods), int(goodsInfo.Num))
      if err != nil {
         txn.Rollback() //回滚
         log.Errorf("订单%s扣减库存失败", ordersn)
         return err
      }

   }

   err := is.data.Inventorys().CreateStockSellDetail(ctx, txn, &sellDetail)
   if err != nil {
      txn.Rollback() //回滚
      log.Errorf("订单%s创建扣减库存记录失败", ordersn)
      return err
   }

   txn.Commit()
   return nil
}


问题已解决,确定采纳
还有疑问,暂不采纳

恭喜解决一个难题,获得1积分~

来为老师/同学的回答评分吧

0 星

相似问题

登录后可查看更多问答,登录/注册

Go开发工程师全新版
  • 参与学习       493    人
  • 解答问题       572    个

风口上的技术,薪资水平遥遥领先,现在学习正值红利期! 未来3-5年,Go语言势必成为企业高性能项目中不可替代的语言 从基础到项目实战再到重构,对转行人员友好,真正从入门到精通!

了解课程
请稍等 ...
意见反馈 帮助中心 APP下载
官方微信

在线咨询

领取优惠

免费试听

领取大纲

扫描二维码,添加
你的专属老师