StockSale.php 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. <?php
  2. namespace addons\shopro\library\traits;
  3. use addons\shopro\exception\Exception;
  4. use addons\shopro\model\ActivityGoodsSkuPrice;
  5. use addons\shopro\model\Goods;
  6. use addons\shopro\model\GoodsSkuPrice;
  7. use addons\shopro\model\OrderItem;
  8. use addons\shopro\model\ScoreGoodsSkuPrice;
  9. /**
  10. * 库存销量
  11. */
  12. trait StockSale
  13. {
  14. use StockWarning;
  15. // cache 正向加销量,添加订单之前拦截
  16. public function cacheForwardSale($goods_list) {
  17. try {
  18. // 记录库存不足,中断的位置
  19. $break_key = -1;
  20. foreach ($goods_list as $key => $goods) {
  21. $detail = $goods['detail'];
  22. $activity = $detail['activity'];
  23. // 没有活动,不是秒杀|拼团,或者没有 redis
  24. if (!$activity || !in_array($activity['type'], ['seckill', 'groupon']) || !$this->hasRedis()) {
  25. continue;
  26. }
  27. // 实例化 redis
  28. $redis = $this->getRedis();
  29. $keys = $this->getKeys([
  30. 'goods_id' => $detail['id'],
  31. 'goods_sku_price_id' => $detail['current_sku_price']['id'],
  32. ], [
  33. 'activity_id' => $activity['id'],
  34. 'activity_type' => $activity['type'],
  35. ]);
  36. extract($keys);
  37. // 活动商品规格
  38. $goodsSkuPrice = $redis->HGET($activityHashKey, $goodsSkuPriceKey);
  39. $goodsSkuPrice = json_decode($goodsSkuPrice, true);
  40. // 活动商品库存
  41. $stock = $goodsSkuPrice['stock'] ?? 0;
  42. // 当前销量 + 购买数量 ,salekey 如果不存在,自动创建
  43. $sale = $redis->HINCRBY($activityHashKey, $saleKey, $goods['goods_num']);
  44. if ($sale > $stock) {
  45. // 记录中断的位置
  46. $break_key = $key;
  47. throw new \Exception('活动商品库存不足');
  48. }
  49. }
  50. } catch (\Exception $e) {
  51. // 将 缓存的 销量减掉
  52. if ($break_key >= 0) {
  53. foreach ($goods_list as $key => $goods) {
  54. if ($key > $break_key) { // 上面库存不足中断的位置
  55. break;
  56. }
  57. $detail = $goods['detail'];
  58. $activity = $detail['activity'];
  59. // 没有活动,不是秒杀|拼团,或者没有 redis
  60. if (!$activity || !in_array($activity['type'], ['seckill', 'groupon']) || !$this->hasRedis()) {
  61. continue;
  62. }
  63. // 实例化 redis
  64. $redis = $this->getRedis();
  65. $keys = $this->getKeys([
  66. 'goods_id' => $detail['id'],
  67. 'goods_sku_price_id' => $detail['current_sku_price']['id'],
  68. ], [
  69. 'activity_id' => $activity['id'],
  70. 'activity_type' => $activity['type'],
  71. ]);
  72. extract($keys);
  73. if ($redis->EXISTS($activityHashKey) && $redis->HEXISTS($activityHashKey, $saleKey)) {
  74. $sale = $redis->HINCRBY($activityHashKey, $saleKey, -$goods['goods_num']);
  75. }
  76. }
  77. new Exception('商品库存不足');
  78. }
  79. new Exception($e->getMessage());
  80. }
  81. }
  82. // cache 反向减销量,取消订单/订单自动关闭 时候
  83. public function cacheBackSale($order) {
  84. $items = OrderItem::where('order_id', $order['id'])->select();
  85. foreach ($items as $key => $item) {
  86. $this->cacheBackSaleByItem($item);
  87. }
  88. }
  89. // 真实正向 减库存加销量(支付成功扣库存,加销量)
  90. public function realForwardStockSale($order) {
  91. $items = OrderItem::where('order_id', $order['id'])->select();
  92. foreach ($items as $key => $orderItem) {
  93. // 增加商品销量
  94. Goods::where('id', $orderItem->goods_id)->setInc('sales', $orderItem->goods_num);
  95. $goodsSkuPrice = GoodsSkuPrice::where('id', $orderItem->goods_sku_price_id)->find();
  96. if ($goodsSkuPrice) {
  97. $goodsSkuPrice->setDec('stock', $orderItem->goods_num); // 减少库存
  98. $goodsSkuPrice->setInc('sales', $orderItem->goods_num); // 增加销量
  99. // 库存预警检测
  100. $this->checkStockWarning($goodsSkuPrice);
  101. }
  102. if ($orderItem->item_goods_sku_price_id) {
  103. if ($order['type'] == 'score') {
  104. // 积分商城商品,扣除积分规格库存
  105. $itemGoodsSkuPrice = ScoreGoodsSkuPrice::where('id', $orderItem->item_goods_sku_price_id)->find();
  106. } else {
  107. // 扣除活动库存
  108. $itemGoodsSkuPrice = ActivityGoodsSkuPrice::where('id', $orderItem->item_goods_sku_price_id)->find();
  109. }
  110. if ($itemGoodsSkuPrice) {
  111. $itemGoodsSkuPrice->setDec('stock', $orderItem->goods_num); // 减少库存
  112. $itemGoodsSkuPrice->setInc('sales', $orderItem->goods_num); // 增加销量
  113. }
  114. }
  115. // 已经真实减库存 减掉预销量,库存都是在缓存中读取的, 真是减库存,没有减掉 缓存库存,所以不需要减掉预销量, 设置开始的活动不可编辑
  116. // $this->cacheBackSaleByItem($orderItem);
  117. }
  118. }
  119. // 真实反向 加库存减销量(支付过的现在没有返库存,暂时没用)
  120. public function realBackStockSale($order)
  121. {
  122. $items = OrderItem::where('order_id', $order['id'])->select();
  123. foreach ($items as $key => $orderItem) {
  124. // 返还商品销量
  125. Goods::where('id', $orderItem->goods_id)->setDec('sales', $orderItem->goods_num);
  126. // 返还规格库存
  127. $goodsSkuPrice = GoodsSkuPrice::where('id', $orderItem->goods_sku_price_id)->find();
  128. if ($goodsSkuPrice) {
  129. $goodsSkuPrice->setInc('stock', $orderItem->goods_num);
  130. $goodsSkuPrice->setDec('sales', $orderItem->goods_num);
  131. }
  132. if ($orderItem->item_goods_sku_price_id) {
  133. if ($order['type'] == 'score') {
  134. // 积分商城商品,扣除积分规格库存
  135. $itemGoodsSkuPrice = ScoreGoodsSkuPrice::where('id', $orderItem->item_goods_sku_price_id)->find();
  136. } else {
  137. $itemGoodsSkuPrice = ActivityGoodsSkuPrice::where('id', $orderItem->item_goods_sku_price_id)->find();
  138. }
  139. if ($itemGoodsSkuPrice) {
  140. $itemGoodsSkuPrice->setInc('stock', $orderItem->goods_num); // 增加库存
  141. $itemGoodsSkuPrice->setDec('sales', $orderItem->goods_num); // 减少销量
  142. }
  143. }
  144. }
  145. }
  146. // 通过 OrderItem 减预库存
  147. private function cacheBackSaleByItem($item)
  148. {
  149. // 不是秒杀|拼团,或者 没有配置 redis
  150. if ((strpos($item['activity_type'], 'groupon') === false && strpos($item['activity_type'], 'seckill') === false) || !$this->hasRedis()) {
  151. return false;
  152. }
  153. // 实例化 redis
  154. $redis = $this->getRedis();
  155. $keys = $this->getKeys([
  156. 'goods_id' => $item['goods_id'],
  157. 'goods_sku_price_id' => $item['goods_sku_price_id'],
  158. ], [
  159. 'activity_id' => $item['activity_id'],
  160. 'activity_type' => $item['activity_type'],
  161. ]);
  162. extract($keys);
  163. if ($redis->EXISTS($activityHashKey) && $redis->HEXISTS($activityHashKey, $saleKey)) {
  164. $sale = $redis->HINCRBY($activityHashKey, $saleKey, -$item['goods_num']);
  165. }
  166. return true;
  167. }
  168. }