RefundLogic.php 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | likeadmin快速开发前后端分离管理后台(PHP版)
  4. // +----------------------------------------------------------------------
  5. // | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
  6. // | 开源版本可自由商用,可去除界面版权logo
  7. // | gitee下载:https://gitee.com/likeshop_gitee/likeadmin
  8. // | github下载:https://github.com/likeshop-github/likeadmin
  9. // | 访问官网:https://www.likeadmin.cn
  10. // | likeadmin团队 版权所有 拥有最终解释权
  11. // +----------------------------------------------------------------------
  12. // | author: likeadminTeam
  13. // +----------------------------------------------------------------------
  14. namespace app\common\logic;
  15. use app\common\enum\PayEnum;
  16. use app\common\enum\RefundEnum;
  17. use app\common\model\recharge\RechargeOrder;
  18. use app\common\model\refund\RefundLog;
  19. use app\common\model\refund\RefundRecord;
  20. use app\common\service\pay\AliPayService;
  21. use app\common\service\pay\WeChatPayService;
  22. /**
  23. * 订单退款逻辑
  24. * Class OrderRefundLogic
  25. * @package app\common\logic
  26. */
  27. class RefundLogic extends BaseLogic
  28. {
  29. protected static $refundLog;
  30. /**
  31. * @notes 发起退款
  32. * @param $order
  33. * @param $refundRecordId
  34. * @param $refundAmount
  35. * @param $handleId
  36. * @return bool
  37. * @throws \Exception
  38. * @author 段誉
  39. * @date 2023/2/28 17:24
  40. */
  41. public static function refund($order, $refundRecordId, $refundAmount, $handleId)
  42. {
  43. // 退款前校验
  44. self::refundBeforeCheck($refundAmount);
  45. // 添加退款日志
  46. self::log($order, $refundRecordId, $refundAmount, $handleId);
  47. // 根据不同支付方式退款
  48. try {
  49. switch ($order['pay_way']) {
  50. //微信退款
  51. case PayEnum::WECHAT_PAY:
  52. self::wechatPayRefund($order, $refundAmount);
  53. break;
  54. // 支付宝退款
  55. case PayEnum::ALI_PAY:
  56. self::aliPayRefund($refundRecordId, $refundAmount);
  57. break;
  58. default:
  59. throw new \Exception('支付方式异常');
  60. }
  61. // 此处true并不表示退款成功,仅表示退款请求成功,具体成功与否由定时任务查询或通过退款回调得知
  62. return true;
  63. } catch (\Exception $e) {
  64. // 退款请求失败,标记退款记录及日志为失败.在退款记录处重新退款
  65. self::$error = $e->getMessage();
  66. self::refundFailHandle($refundRecordId, $e->getMessage());
  67. return false;
  68. }
  69. }
  70. /**
  71. * @notes 退款前校验
  72. * @param $refundAmount
  73. * @throws \Exception
  74. * @author 段誉
  75. * @date 2023/2/28 16:27
  76. */
  77. public static function refundBeforeCheck($refundAmount)
  78. {
  79. if ($refundAmount <= 0) {
  80. throw new \Exception('订单金额异常');
  81. }
  82. }
  83. /**
  84. * @notes 微信支付退款
  85. * @param $order
  86. * @param $refundAmount
  87. * @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException
  88. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  89. * @author 段誉
  90. * @date 2023/2/28 17:19
  91. */
  92. public static function wechatPayRefund($order, $refundAmount)
  93. {
  94. // 发起退款。 若发起退款请求返回明确错误,退款日志和记录标记状态为退款失败
  95. // 退款日志及记录的成功状态目前统一由定时任务查询退款结果为退款成功后才标记成功
  96. // 也可通过设置退款回调,在退款回调时处理退款记录状态为成功
  97. (new WeChatPayService($order['order_terminal']))->refund([
  98. 'transaction_id' => $order['transaction_id'],
  99. 'refund_sn' => self::$refundLog['sn'],
  100. 'refund_amount' => $refundAmount,// 退款金额
  101. 'total_amount' => $order['order_amount'],// 订单金额
  102. ]);
  103. }
  104. /**
  105. * @notes 支付宝退款
  106. * @param $refundRecordId
  107. * @param $refundAmount
  108. * @throws \Exception
  109. * @author mjf
  110. * @date 2024/3/18 18:54
  111. */
  112. public static function aliPayRefund($refundRecordId, $refundAmount)
  113. {
  114. $refundRecord = RefundRecord::where('id', $refundRecordId)->findOrEmpty()->toArray();
  115. $result = (new AliPayService())->refund($refundRecord['order_sn'], $refundAmount, self::$refundLog['sn']);
  116. $result = (array)$result;
  117. if ($result['code'] == '10000' && $result['msg'] == 'Success' && $result['fundChange'] == 'Y') {
  118. // 更新日志
  119. RefundLog::update([
  120. 'refund_status' => RefundEnum::REFUND_SUCCESS,
  121. 'refund_msg' => json_encode($result, JSON_UNESCAPED_UNICODE),
  122. ], ['id'=>self::$refundLog['id']]);
  123. // 更新记录
  124. RefundRecord::update([
  125. 'refund_status' => RefundEnum::REFUND_SUCCESS,
  126. ], ['id'=>$refundRecordId]);
  127. // 更新订单信息
  128. if ($refundRecord['order_type'] == 'recharge') {
  129. RechargeOrder::update([
  130. 'id' => $refundRecord['order_id'],
  131. 'refund_transaction_id' => $result['tradeNo'] ?? '',
  132. ]);
  133. }
  134. }
  135. }
  136. /**
  137. * @notes 退款请求失败处理
  138. * @param $refundRecordId
  139. * @author 段誉
  140. * @date 2023/2/28 16:00
  141. * @remark 【微信,支付宝】退款接口请求失败时,更新退款记录及日志为失败,在退款记录重新发起
  142. */
  143. public static function refundFailHandle($refundRecordId, $msg)
  144. {
  145. // 更新退款日志记录
  146. RefundLog::update([
  147. 'id' => self::$refundLog['id'],
  148. 'refund_status' => RefundEnum::REFUND_ERROR,
  149. 'refund_msg' => $msg,
  150. ]);
  151. // 更新退款记录状态为退款失败
  152. RefundRecord::update([
  153. 'id' => $refundRecordId,
  154. 'refund_status' => RefundEnum::REFUND_ERROR,
  155. 'refund_msg' => $msg,
  156. ]);
  157. }
  158. /**
  159. * @notes 退款日志
  160. * @param $order
  161. * @param $refundRecordId
  162. * @param $refundAmount
  163. * @param $handleId
  164. * @param int $refundStatus
  165. * @author 段誉
  166. * @date 2023/2/28 15:29
  167. */
  168. public static function log($order, $refundRecordId, $refundAmount, $handleId, $refundStatus = RefundEnum::REFUND_ING)
  169. {
  170. $sn = generate_sn(RefundLog::class, 'sn');
  171. self::$refundLog = RefundLog::create([
  172. 'sn' => $sn,
  173. 'record_id' => $refundRecordId,
  174. 'user_id' => $order['user_id'],
  175. 'handle_id' => $handleId,
  176. 'order_amount' => $order['order_amount'],
  177. 'refund_amount' => $refundAmount,
  178. 'refund_status' => $refundStatus
  179. ]);
  180. }
  181. }