RefundLogic.php 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | LikeShop有特色的全开源社交分销电商系统
  4. // +----------------------------------------------------------------------
  5. // | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
  6. // | 商业用途务必购买系统授权,以免引起不必要的法律纠纷
  7. // | 禁止对系统程序代码以任何目的,任何形式的再发布
  8. // | 微信公众号:好象科技
  9. // | 访问官网:http://www.likemarket.net
  10. // | 访问社区:http://bbs.likemarket.net
  11. // | 访问手册:http://doc.likemarket.net
  12. // | 好象科技开发团队 版权所有 拥有最终解释权
  13. // +----------------------------------------------------------------------
  14. // | Author: LikeShopTeam-段誉
  15. // +----------------------------------------------------------------------
  16. namespace app\common\logic;
  17. use app\common\enum\AfterSaleEnum;
  18. use app\common\enum\AfterSaleLogEnum;
  19. use app\common\enum\DeliveryEnum;
  20. use app\common\enum\OrderEnum;
  21. use app\common\enum\PayEnum;
  22. use app\common\model\AfterSale;
  23. use app\common\model\Order;
  24. use app\common\model\OrderGoods;
  25. use app\common\model\Refund;
  26. use app\common\service\after_sale\AfterSaleService;
  27. use app\common\service\pay\AliPayService;
  28. use app\common\service\pay\BalancePayService;
  29. use app\common\service\pay\ToutiaoPayService;
  30. use app\common\service\pay\WeChatPayService;
  31. use app\common\service\WeChatConfigService;
  32. use think\db\exception\DataNotFoundException;
  33. use think\db\exception\DbException;
  34. use think\db\exception\ModelNotFoundException;
  35. /**
  36. * 订单退款逻辑
  37. * Class OrderRefundLogic
  38. * @package app\common\logic
  39. */
  40. class RefundLogic extends BaseLogic
  41. {
  42. protected static $refund;
  43. /**
  44. * @notes 发起退款
  45. * @param $refundWay //退款类型;(原路退,退回到余额)
  46. * @param $order //订单信息
  47. * @param $afterSaleId //售后退款id
  48. * @param $refundAmount //退款金额
  49. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  50. * @author 段誉
  51. * @date 2021/8/9 17:31
  52. */
  53. public static function refund($refundWay, $order, $afterSaleId, $refundAmount)
  54. {
  55. if ($refundAmount < 0) {
  56. return false;
  57. }
  58. self::log($order, $afterSaleId, $refundAmount);
  59. if ($refundAmount == 0) {
  60. //更新退款日志记录
  61. Refund::update([
  62. 'refund_status' => 1
  63. ], ['id' => self::$refund['id']]);
  64. //更新售后状态
  65. $afterSale = AfterSale::findOrEmpty($afterSaleId);
  66. // 整单退款
  67. if($afterSale['refund_type'] == AfterSaleEnum::REFUND_TYPE_ORDER) {
  68. $order = Order::findOrEmpty($afterSale['order_id'])->toArray();
  69. $refundStauts = ($afterSale['refund_total_amount'] == $order['order_amount']) ? AfterSaleEnum::FULL_REFUND : AfterSaleEnum::PARTIAL_REFUND;
  70. }
  71. // 商品售后
  72. if($afterSale['refund_type'] == AfterSaleEnum::REFUND_TYPE_GOODS) {
  73. $orderGoods = OrderGoods::findOrEmpty($afterSale['order_goods_id'])->toArray();
  74. $refundStauts = ($afterSale['refund_total_amount'] == $orderGoods['total_pay_price']) ? AfterSaleEnum::FULL_REFUND : AfterSaleEnum::PARTIAL_REFUND;
  75. }
  76. $afterSale->status = AfterSaleEnum::STATUS_SUCCESS;
  77. $afterSale->sub_status = AfterSaleEnum::SUB_STATUS_SELLER_REFUND_SUCCESS;
  78. $afterSale->refund_status = $refundStauts ?? AfterSaleEnum::FULL_REFUND;
  79. $afterSale->save();
  80. AfterSaleService::createAfterLog($afterSale['id'], '系统已完成退款', 0, AfterSaleLogEnum::ROLE_SYS);
  81. return true;
  82. }
  83. //区分原路退 还是退回到余额
  84. if ($refundWay == AfterSaleEnum::REFUND_WAYS_BALANCE) {
  85. self::balancePayRefund($order, $refundAmount, $afterSaleId);
  86. return true;
  87. }
  88. switch ($order['pay_way']) {
  89. //余额退款
  90. case PayEnum::BALANCE_PAY:
  91. self::balancePayRefund($order, $refundAmount,$afterSaleId);
  92. break;
  93. //微信退款
  94. case PayEnum::WECHAT_PAY:
  95. self::wechatPayRefund($order, $refundAmount);
  96. break;
  97. //支付宝退款
  98. case PayEnum::ALI_PAY:
  99. self::aliPayRefund($order, $refundAmount);
  100. break;
  101. //字节退款
  102. case PayEnum::BYTE_PAY:
  103. self::bytePayRefund($order, $refundAmount);
  104. break;
  105. // 线下支付 直接完成
  106. case PayEnum::OFFLINE_PAY:
  107. return true;
  108. break;
  109. }
  110. return true;
  111. }
  112. /**
  113. * @notes 余额退款
  114. * @param $order
  115. * @param $refundAmount
  116. * @author 段誉
  117. * @date 2021/8/5 10:25
  118. */
  119. public static function balancePayRefund($order, $refundAmount,$afterSaleId)
  120. {
  121. (new BalancePayService())->refund($order, $refundAmount,$afterSaleId);
  122. }
  123. /**
  124. * @notes 微信退款
  125. * @param $refundWay
  126. * @param $order
  127. * @param $refundAmount
  128. * @return bool|void
  129. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  130. * @author 段誉
  131. * @date 2021/8/5 10:25
  132. */
  133. public static function wechatPayRefund($order, $refundAmount)
  134. {
  135. $pay = new WeChatPayService($order['order_terminal']);
  136. $result = $pay->refund([
  137. 'transaction_id' => $order['transaction_id'],
  138. 'refund_sn' => self::$refund['sn'],
  139. 'total_fee' => $order['order_amount'],
  140. 'refund_fee' => $refundAmount,
  141. ]);
  142. if ($result !== true) {
  143. throw new \Exception($pay->realPay()->getMessage());
  144. }
  145. //更新退款日志记录
  146. Refund::update([
  147. 'wechat_refund_id' => 0,
  148. 'refund_status' => 1,
  149. 'refund_msg' => json_encode([], JSON_UNESCAPED_UNICODE),
  150. ], ['id' => self::$refund['id']]);
  151. }
  152. /**
  153. * @notes 支付宝退款
  154. * @param $order
  155. * @param $refundAmount
  156. * @return bool
  157. * @throws \Exception
  158. * @author 段誉
  159. * @date 2021/8/5 10:25
  160. */
  161. public static function aliPayRefund($order, $refundAmount)
  162. {
  163. //原路退回到支付宝的情况
  164. $result = (new AliPayService())->refund($order['sn'], $refundAmount, self::$refund['sn']);
  165. $result = (array)$result;
  166. //更新退款日志记录
  167. Refund::update([
  168. 'refund_status' => (isset($result['result_code']) && $result['result_code'] == 'SUCCESS') ? 1 : 2,
  169. 'refund_msg' => json_encode($result['httpBody'], JSON_UNESCAPED_UNICODE),
  170. ], ['id' => self::$refund['id']]);
  171. if ($result['code'] != '10000' || $result['msg'] != 'Success' || $result['fundChange'] != 'Y') {
  172. throw new \Exception('支付宝退款失败');
  173. }
  174. return true;
  175. }
  176. /**
  177. * @notes 字节退款
  178. * @param $order
  179. * @param $refundAmount
  180. * @author Tab
  181. * @date 2021/11/18 14:09
  182. */
  183. public static function bytePayRefund($order, $refundAmount)
  184. {
  185. (new ToutiaoPayService())->refund($order, $refundAmount, self::$refund);
  186. }
  187. /**
  188. * @notes 退款日志
  189. * @param $order
  190. * @param $afterSaleId
  191. * @param $refundAmount
  192. * @author 段誉
  193. * @date 2021/8/9 17:32
  194. */
  195. public static function log($order, $afterSaleId, $refundAmount)
  196. {
  197. $result = Refund::create([
  198. 'order_id' => $order['id'],
  199. 'after_sale_id' => $afterSaleId,
  200. 'user_id' => $order['user_id'],
  201. 'sn' => generate_sn(new Refund(), 'sn'),
  202. 'order_amount' => $order['order_amount'],
  203. 'refund_amount' => $refundAmount,
  204. ]);
  205. self::$refund = $result;
  206. }
  207. /**
  208. * @notes 售后退款更新订单或订单商品状态
  209. * @param $order_id
  210. * @return void
  211. * @throws DbException
  212. * @throws ModelNotFoundException
  213. * @throws DataNotFoundException
  214. * @author lbzy
  215. * @datetime 2023-07-07 13:53:19
  216. */
  217. static function afterSaleRefundUpdate($order_id): void
  218. {
  219. $order_goods_count = OrderGoods::where('order_id', $order_id)->count();
  220. $after_sale_count = AfterSale::where('order_id', $order_id)
  221. ->group('order_goods_id')
  222. ->where('status', AfterSaleEnum::STATUS_SUCCESS)
  223. ->count();
  224. //如果订单商品已全部退款
  225. if ($order_goods_count == $after_sale_count) {
  226. Order::update([ 'order_status' => OrderEnum::STATUS_CLOSE ], [
  227. [ 'id', '=', $order_id ],
  228. ]);
  229. } else {
  230. // 更新订单发货状态
  231. $orderGoodsIds = OrderGoods::where(['order_id'=>$order_id,'express_status'=>[DeliveryEnum::NOT_SHIPPED,DeliveryEnum::PART_SHIPPED]])->column('id');
  232. if (!empty($orderGoodsIds)) {
  233. foreach ($orderGoodsIds as $key=>$orderGoodsId) {
  234. $afterSale = AfterSale::where(['order_goods_id'=>$orderGoodsId,'status'=>AfterSaleEnum::STATUS_SUCCESS])->findOrEmpty();
  235. if (!$afterSale->isEmpty()) {
  236. unset($orderGoodsIds[$key]);
  237. }
  238. }
  239. }
  240. if (empty($orderGoodsIds)) {
  241. $order = Order::find($order_id);
  242. $order->order_status = max($order->order_status, OrderEnum::STATUS_WAIT_RECEIVE);
  243. $order->express_status = DeliveryEnum::SHIPPED;
  244. $order->save();
  245. }
  246. }
  247. }
  248. }