RefundLogic.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  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\service\pay\MixedPayService;
  18. use app\common\enum\AfterSaleEnum;
  19. use app\common\enum\AfterSaleLogEnum;
  20. use app\common\enum\DeliveryEnum;
  21. use app\common\enum\OrderEnum;
  22. use app\common\enum\PayEnum;
  23. use app\common\model\AfterSale;
  24. use app\common\model\Order;
  25. use app\common\model\OrderGoods;
  26. use app\common\model\Refund;
  27. use app\common\service\after_sale\AfterSaleService;
  28. use app\common\service\pay\AliPayService;
  29. use app\common\service\pay\BalancePayService;
  30. use app\common\service\pay\ToutiaoPayService;
  31. use app\common\service\pay\WeChatPayService;
  32. use app\common\service\WeChatConfigService;
  33. use think\db\exception\DataNotFoundException;
  34. use think\db\exception\DbException;
  35. use think\db\exception\ModelNotFoundException;
  36. /**
  37. * 订单退款逻辑
  38. * Class OrderRefundLogic
  39. * @package app\common\logic
  40. */
  41. class RefundLogic extends BaseLogic
  42. {
  43. protected static $refund;
  44. /**
  45. * @notes 发起退款
  46. * @param $refundWay //退款类型;(原路退,退回到余额)
  47. * @param $order //订单信息
  48. * @param $afterSaleId //售后退款id
  49. * @param $refundAmount //退款金额
  50. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  51. * @author 段誉
  52. * @date 2021/8/9 17:31
  53. */
  54. public static function refund($refundWay, $order, $afterSaleId, $refundAmount)
  55. {
  56. outFileLog($refundWay,'refund_money','$refundWay');
  57. outFileLog($afterSaleId,'refund_money','$afterSaleId');
  58. outFileLog($refundAmount,'refund_money','$refundAmount');
  59. if ($refundAmount < 0) {
  60. return false;
  61. }
  62. self::log($order, $afterSaleId, $refundAmount);
  63. if ($refundAmount == 0) {
  64. //更新退款日志记录
  65. Refund::update([
  66. 'refund_status' => 1
  67. ], ['id' => self::$refund['id']]);
  68. //更新售后状态
  69. $afterSale = AfterSale::findOrEmpty($afterSaleId);
  70. // 整单退款
  71. if($afterSale['refund_type'] == AfterSaleEnum::REFUND_TYPE_ORDER) {
  72. $order = Order::findOrEmpty($afterSale['order_id'])->toArray();
  73. $refundStauts = ($afterSale['refund_total_amount'] == $order['order_amount']) ? AfterSaleEnum::FULL_REFUND : AfterSaleEnum::PARTIAL_REFUND;
  74. }
  75. // 商品售后
  76. if($afterSale['refund_type'] == AfterSaleEnum::REFUND_TYPE_GOODS) {
  77. $orderGoods = OrderGoods::findOrEmpty($afterSale['order_goods_id'])->toArray();
  78. $refundStauts = ($afterSale['refund_total_amount'] == $orderGoods['total_pay_price']) ? AfterSaleEnum::FULL_REFUND : AfterSaleEnum::PARTIAL_REFUND;
  79. }
  80. $afterSale->status = AfterSaleEnum::STATUS_SUCCESS;
  81. $afterSale->sub_status = AfterSaleEnum::SUB_STATUS_SELLER_REFUND_SUCCESS;
  82. $afterSale->refund_status = $refundStauts ?? AfterSaleEnum::FULL_REFUND;
  83. $afterSale->save();
  84. AfterSaleService::createAfterLog($afterSale['id'], '系统已完成退款', 0, AfterSaleLogEnum::ROLE_SYS);
  85. return true;
  86. }
  87. //区分原路退 还是退回到余额
  88. if ($refundWay == AfterSaleEnum::REFUND_WAYS_BALANCE) {
  89. self::balancePayRefund($order, $refundAmount, $afterSaleId);
  90. return true;
  91. }
  92. switch ($order['pay_way']) {
  93. //余额退款
  94. case PayEnum::BALANCE_PAY:
  95. self::balancePayRefund($order, $refundAmount,$afterSaleId);
  96. break;
  97. //微信退款
  98. case PayEnum::WECHAT_PAY:
  99. self::wechatPayRefund($order, $refundAmount);
  100. break;
  101. //支付宝退款
  102. case PayEnum::ALI_PAY:
  103. self::aliPayRefund($order, $refundAmount);
  104. break;
  105. //字节退款
  106. case PayEnum::BYTE_PAY:
  107. self::bytePayRefund($order, $refundAmount);
  108. break;
  109. //混合支付退款
  110. case PayEnum::MIXED_PAY:
  111. self::mixedPayRefund($order, $refundAmount, $afterSaleId);
  112. break;
  113. // 线下支付 直接完成
  114. case PayEnum::OFFLINE_PAY:
  115. return true;
  116. break;
  117. }
  118. return true;
  119. }
  120. /**
  121. * @notes 余额退款
  122. * @param $order
  123. * @param $refundAmount
  124. * @author 段誉
  125. * @date 2021/8/5 10:25
  126. */
  127. public static function balancePayRefund($order, $refundAmount,$afterSaleId)
  128. {
  129. (new BalancePayService())->refund($order, $refundAmount,$afterSaleId);
  130. }
  131. /**
  132. * @notes 微信退款
  133. * @param $refundWay
  134. * @param $order
  135. * @param $refundAmount
  136. * @return bool|void
  137. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  138. * @author 段誉
  139. * @date 2021/8/5 10:25
  140. */
  141. public static function wechatPayRefund($order, $refundAmount)
  142. {
  143. $pay = new WeChatPayService($order['order_terminal']);
  144. $result = $pay->refund([
  145. 'transaction_id' => $order['transaction_id'],
  146. 'refund_sn' => self::$refund['sn'],
  147. 'total_fee' => $order['order_amount'],
  148. 'refund_fee' => $refundAmount,
  149. ]);
  150. if ($result !== true) {
  151. throw new \Exception($pay->realPay()->getMessage());
  152. }
  153. //更新退款日志记录
  154. Refund::update([
  155. 'wechat_refund_id' => 0,
  156. 'refund_status' => 1,
  157. 'refund_msg' => json_encode([], JSON_UNESCAPED_UNICODE),
  158. ], ['id' => self::$refund['id']]);
  159. //更新售后订单状态
  160. $re_where['order_id'] = $order['id'];
  161. $re_where['refund_total_amount'] = $refundAmount;
  162. $afterSale = AfterSale::where($re_where)->findOrEmpty();
  163. if(!$afterSale->isEmpty()){
  164. $afterSale->status = AfterSaleEnum::STATUS_SUCCESS;
  165. $afterSale->sub_status = AfterSaleEnum::SUB_STATUS_SELLER_REFUND_SUCCESS;
  166. $afterSale->refund_status = $refundStauts ?? AfterSaleEnum::FULL_REFUND;
  167. $afterSale->save();
  168. AfterSaleService::createAfterLog($afterSale['id'], '系统已完成退款', 0, AfterSaleLogEnum::ROLE_SYS);
  169. }
  170. }
  171. /**
  172. * @notes 支付宝退款
  173. * @param $order
  174. * @param $refundAmount
  175. * @return bool
  176. * @throws \Exception
  177. * @author 段誉
  178. * @date 2021/8/5 10:25
  179. */
  180. public static function aliPayRefund($order, $refundAmount)
  181. {
  182. //原路退回到支付宝的情况
  183. $result = (new AliPayService())->refund($order['sn'], $refundAmount, self::$refund['sn']);
  184. $result = (array)$result;
  185. //更新退款日志记录
  186. Refund::update([
  187. 'refund_status' => (isset($result['result_code']) && $result['result_code'] == 'SUCCESS') ? 1 : 2,
  188. 'refund_msg' => json_encode($result['httpBody'], JSON_UNESCAPED_UNICODE),
  189. ], ['id' => self::$refund['id']]);
  190. if ($result['code'] != '10000' || $result['msg'] != 'Success' || $result['fundChange'] != 'Y') {
  191. throw new \Exception('支付宝退款失败');
  192. }
  193. return true;
  194. }
  195. /**
  196. * @notes 字节退款
  197. * @param $order
  198. * @param $refundAmount
  199. * @author Tab
  200. * @date 2021/11/18 14:09
  201. */
  202. public static function bytePayRefund($order, $refundAmount)
  203. {
  204. (new ToutiaoPayService())->refund($order, $refundAmount, self::$refund);
  205. }
  206. /**
  207. * @notes 混合支付退款
  208. * @param $order 订单信息
  209. * @param $refundAmount 退款金额
  210. * @param $afterSaleId 售后ID
  211. * @return bool
  212. * @author 系统
  213. * @date 2024/12/19
  214. */
  215. public static function mixedPayRefund($order, $refundAmount, $afterSaleId)
  216. {
  217. try {
  218. $mixedPayServiceModel = new MixedPayService($order['order_terminal'], $order['user_id']);
  219. $result = $mixedPayServiceModel->refund($order, $refundAmount, $afterSaleId);
  220. if ($result === false) {
  221. throw new \Exception('混合支付退款失败:' . $mixedPayServiceModel->getError());
  222. }
  223. return true;
  224. } catch (\Exception $e) {
  225. throw new \Exception('混合支付退款处理失败:' . $e->getMessage());
  226. }
  227. }
  228. /**
  229. * @notes 退款日志
  230. * @param $order
  231. * @param $afterSaleId
  232. * @param $refundAmount
  233. * @author 段誉
  234. * @date 2021/8/9 17:32
  235. */
  236. public static function log($order, $afterSaleId, $refundAmount)
  237. {
  238. $result = Refund::create([
  239. 'order_id' => $order['id'],
  240. 'after_sale_id' => $afterSaleId,
  241. 'user_id' => $order['user_id'],
  242. 'sn' => generate_sn(new Refund(), 'sn'),
  243. 'order_amount' => $order['order_amount'],
  244. 'refund_amount' => $refundAmount,
  245. ]);
  246. self::$refund = $result;
  247. }
  248. /**
  249. * @notes 售后退款更新订单或订单商品状态
  250. * @param $order_id
  251. * @return void
  252. * @throws DbException
  253. * @throws ModelNotFoundException
  254. * @throws DataNotFoundException
  255. * @author lbzy
  256. * @datetime 2023-07-07 13:53:19
  257. */
  258. static function afterSaleRefundUpdate($order_id): void
  259. {
  260. $order_goods_count = OrderGoods::where('order_id', $order_id)->count();
  261. $after_sale_count = AfterSale::where('order_id', $order_id)
  262. ->group('order_goods_id')
  263. ->where('status', AfterSaleEnum::STATUS_SUCCESS)
  264. ->count();
  265. outFileLog($order_id,'refund_money','afterSaleRefundUpdate-refund');
  266. //如果订单商品已全部退款
  267. if ($order_goods_count == $after_sale_count) {
  268. Order::update([ 'order_status' => OrderEnum::STATUS_CLOSE ], [
  269. [ 'id', '=', $order_id ],
  270. ]);
  271. } else {
  272. // 更新订单发货状态
  273. $orderGoodsIds = OrderGoods::where(['order_id'=>$order_id,'express_status'=>[DeliveryEnum::NOT_SHIPPED,DeliveryEnum::PART_SHIPPED]])->column('id');
  274. if (!empty($orderGoodsIds)) {
  275. foreach ($orderGoodsIds as $key=>$orderGoodsId) {
  276. $afterSale = AfterSale::where(['order_goods_id'=>$orderGoodsId,'status'=>AfterSaleEnum::STATUS_SUCCESS])->findOrEmpty();
  277. if (!$afterSale->isEmpty()) {
  278. unset($orderGoodsIds[$key]);
  279. }
  280. }
  281. }
  282. if (empty($orderGoodsIds)) {
  283. $order = Order::find($order_id);
  284. $order->order_status = max($order->order_status, OrderEnum::STATUS_WAIT_RECEIVE);
  285. $order->express_status = DeliveryEnum::SHIPPED;
  286. $order->save();
  287. }
  288. }
  289. }
  290. }