截流自动化的商城平台
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

IntegralOrderRefundLogic.php 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | likeshop开源商城系统
  4. // +----------------------------------------------------------------------
  5. // | 欢迎阅读学习系统程序代码,建议反馈是我们前进的动力
  6. // | gitee下载:https://gitee.com/likeshop_gitee
  7. // | github下载:https://github.com/likeshop-github
  8. // | 访问官网:https://www.likeshop.cn
  9. // | 访问社区:https://home.likeshop.cn
  10. // | 访问手册:http://doc.likeshop.cn
  11. // | 微信公众号:likeshop技术社区
  12. // | likeshop系列产品在gitee、github等公开渠道开源版本可免费商用,未经许可不能去除前后端官方版权标识
  13. // | likeshop系列产品收费版本务必购买商业授权,购买去版权授权后,方可去除前后端官方版权标识
  14. // | 禁止对系统程序代码以任何目的,任何形式的再发布
  15. // | likeshop团队版权所有并拥有最终解释权
  16. // +----------------------------------------------------------------------
  17. // | author: likeshop.cn.team
  18. // +----------------------------------------------------------------------
  19. namespace app\common\logic;
  20. use app\common\enum\IntegralGoodsEnum;
  21. use app\common\enum\IntegralOrderEnum;
  22. use app\common\enum\PayEnum;
  23. use app\common\model\integral\IntegralGoods;
  24. use app\common\model\integral\IntegralOrder;
  25. use app\common\model\AccountLog;
  26. use app\common\model\integral\IntegralOrderRefund;
  27. use app\common\model\user\User;
  28. use app\common\server\AliPayServer;
  29. use app\common\server\DouGong\pay\PayZhengsaoRefund;
  30. use app\common\server\WeChatPayServer;
  31. use app\common\server\WeChatServer;
  32. use think\Exception;
  33. /**
  34. * 积分订单退款逻辑
  35. * Class OrderRefundLogic
  36. * @package app\common\logic
  37. */
  38. class IntegralOrderRefundLogic
  39. {
  40. /**
  41. * @notes 取消订单(标记订单状态,退回库存,扣减销量)
  42. * @param int $order_id
  43. * @author 段誉
  44. * @date 2022/3/3 11:01
  45. */
  46. public static function cancelOrder(int $order_id)
  47. {
  48. // 订单信息
  49. $order = IntegralOrder::findOrEmpty($order_id);
  50. $order->cancel_time = time();
  51. $order->order_status = IntegralOrderEnum::ORDER_STATUS_DOWN;
  52. $order->save();
  53. // 订单商品信息
  54. $goods_snap = $order['goods_snap'];
  55. // 退回库存, 扣减销量
  56. IntegralGoods::where([['id', '=', $goods_snap['id']], ['sales', '>=', $order['total_num']]])
  57. ->inc('stock', $order['total_num'])
  58. ->dec('sales', $order['total_num'])
  59. ->update();
  60. }
  61. /**
  62. * @notes 退回已支付金额
  63. * @param int $order_id
  64. * @return bool
  65. * @throws Exception
  66. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  67. * @throws \think\db\exception\DataNotFoundException
  68. * @throws \think\db\exception\DbException
  69. * @throws \think\db\exception\ModelNotFoundException
  70. * @author 段誉
  71. * @date 2022/3/3 16:00
  72. */
  73. public static function refundOrderAmount(int $order_id)
  74. {
  75. // 订单信息
  76. $order = IntegralOrder::findOrEmpty($order_id);
  77. // 订单商品信息
  78. $goods_snap = $order['goods_snap'];
  79. //已支付的商品订单,取消,退款
  80. if ($goods_snap['type'] == IntegralGoodsEnum::TYPE_GOODS
  81. && $order['refund_status'] == IntegralOrderEnum::NO_REFUND
  82. ) {
  83. if ($order['order_amount'] <= 0) {
  84. return true;
  85. }
  86. // 更新订单退款状态为已退款
  87. IntegralOrder::where(['id' => $order['id']])->update([
  88. 'refund_status' => IntegralOrderEnum::IS_REFUND,//订单退款状态; 0-未退款;1-已退款
  89. 'refund_amount' => $order['order_amount'],
  90. ]);
  91. // 发起退款
  92. $refund_log = self::addRefundLog($order, $order['order_amount'], 1, $order['order_amount']);
  93. switch ($order['pay_way']) {
  94. //余额退款
  95. case PayEnum::BALANCE_PAY:
  96. self::balancePayRefund($order, $order['order_amount']);
  97. break;
  98. //微信退款
  99. case PayEnum::WECHAT_PAY:
  100. self::wechatPayRefund($order, $refund_log);
  101. break;
  102. //支付宝退款
  103. case PayEnum::ALI_PAY:
  104. self::aliPayRefund($order, $refund_log);
  105. break;
  106. case PayEnum::HFDG_WECHAT:
  107. case PayEnum::HFDG_ALIPAY:
  108. $payZsRefund = new PayZhengsaoRefund([
  109. 'refund' => [
  110. 'id' => $refund_log['id'],
  111. 'money' => $order['order_amount'],
  112. ],
  113. 'order' => [
  114. 'id' => $order['id'],
  115. 'transaction_id' => $order['transaction_id'],
  116. 'hfdg_params' => $order['hfdg_params'],
  117. ],
  118. 'from' => 'integral',
  119. ]);
  120. $result = $payZsRefund->request()->getRefundResult();
  121. if ($result['code'] != 1) {
  122. throw new \Exception($result['msg']);
  123. }
  124. break;
  125. }
  126. }
  127. return true;
  128. }
  129. /**
  130. * @notes 退回已支付积分
  131. * @param $id
  132. * @return bool
  133. * @throws \think\db\exception\DataNotFoundException
  134. * @throws \think\db\exception\ModelNotFoundException
  135. * @author 段誉
  136. * @date 2022/3/3 16:02
  137. */
  138. public static function refundOrderIntegral($id)
  139. {
  140. $order = IntegralOrder::findOrEmpty($id);
  141. if ($order['order_integral'] > 0) {
  142. // 退回积分
  143. User::where(['id' => $order['user_id']])
  144. ->inc('user_integral', $order['order_integral'])
  145. ->update();
  146. AccountLogLogic::AccountRecord(
  147. $order['user_id'],
  148. $order['order_integral'], 1,
  149. AccountLog::cancel_integral_order,
  150. '', $order['id'], $order['order_sn']
  151. );
  152. IntegralOrder::where(['id' => $id])->update([
  153. 'refund_integral' => $order['order_integral']
  154. ]);
  155. }
  156. return true;
  157. }
  158. /**
  159. * @notes 增加退款记录
  160. * @param $order
  161. * @param $refund_amount
  162. * @param $status
  163. * @param string $msg
  164. * @return IntegralOrderRefund|\think\Model
  165. * @throws \think\db\exception\DataNotFoundException
  166. * @throws \think\db\exception\DbException
  167. * @throws \think\db\exception\ModelNotFoundException
  168. * @author 段誉
  169. * @date 2022/3/3 14:51
  170. */
  171. public static function addRefundLog($order, $refund_amount, $status, $msg = '')
  172. {
  173. return IntegralOrderRefund::create([
  174. 'order_id' => $order['id'],
  175. 'user_id' => $order['user_id'],
  176. 'refund_sn' => createSn('order_refund', 'refund_sn'),
  177. 'order_amount' => $order['order_amount'],
  178. 'refund_amount' => $refund_amount,
  179. 'transaction_id' => $order['transaction_id'],
  180. 'create_time' => time(),
  181. 'refund_status' => $status,
  182. 'refund_at' => time(),
  183. 'refund_msg' => json_encode($msg, JSON_UNESCAPED_UNICODE),
  184. ]);
  185. }
  186. /**
  187. * @notes 微信支付退款
  188. * @param $order
  189. * @param $refund_id
  190. * @throws Exception
  191. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  192. * @author 段誉
  193. * @date 2022/3/3 14:52
  194. */
  195. public static function wechatPayRefund($order, $refund)
  196. {
  197. $config = WeChatServer::getPayConfigBySource($order['order_source'])['config'];
  198. if (empty($config)) {
  199. throw new Exception('请联系管理员设置微信相关配置!');
  200. }
  201. if (!isset($config['cert_path']) || !isset($config['key_path'])) {
  202. throw new Exception('请联系管理员设置微信证书!');
  203. }
  204. if (!file_exists($config['cert_path']) || !file_exists($config['key_path'])) {
  205. throw new Exception('微信证书不存在,请联系管理员!');
  206. }
  207. $result = WeChatPayServer::refund($config, [
  208. 'transaction_id' => $order['transaction_id'],
  209. 'refund_sn' => $refund['refund_sn'],
  210. 'total_fee' => intval(strval($refund['order_amount'] * 100)),//订单金额,单位为分
  211. 'refund_fee' => intval(strval($refund['refund_amount'] * 100)),//退款金额
  212. ]);
  213. if (isset($result['return_code']) && $result['return_code'] == 'FAIL') {
  214. throw new Exception($result['return_msg']);
  215. }
  216. if (isset($result['err_code_des'])) {
  217. throw new Exception($result['err_code_des']);
  218. }
  219. if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
  220. //更新退款日志记录
  221. IntegralOrderRefund::where(['id' => $refund['id']])->update([
  222. 'wechat_refund_id' => $result['refund_id'] ?? 0,
  223. 'refund_msg' => json_encode($result, JSON_UNESCAPED_UNICODE),
  224. ]);
  225. } else {
  226. throw new Exception('微信支付退款失败');
  227. }
  228. }
  229. /**
  230. * @notes 支付宝退款
  231. * @param $order
  232. * @param $refund_id
  233. * @throws Exception
  234. * @author 段誉
  235. * @date 2022/3/3 14:52
  236. */
  237. public static function aliPayRefund($order, $refund)
  238. {
  239. $result = (new AliPayServer())->refund($order['order_sn'], $order['order_amount']);
  240. // $result = (array)$result;
  241. if ($result['code'] == '10000' && $result['msg'] == 'Success' && $result['fund_change'] == 'Y') {
  242. //更新退款日志记录
  243. IntegralOrderRefund::where(['id' => $refund])->update([
  244. 'refund_msg' => json_encode($result, JSON_UNESCAPED_UNICODE),
  245. ]);
  246. } else {
  247. throw new Exception('支付宝退款失败');
  248. }
  249. }
  250. /**
  251. * @notes 余额退款
  252. * @param $order
  253. * @param $refund_amount
  254. * @return bool
  255. * @throws \think\db\exception\DataNotFoundException
  256. * @throws \think\db\exception\DbException
  257. * @throws \think\db\exception\ModelNotFoundException
  258. * @author 段誉
  259. * @date 2022/3/3 14:52
  260. */
  261. public static function balancePayRefund($order, $refund_amount)
  262. {
  263. $user = User::find($order['user_id']);
  264. $user->user_money = ['inc', $refund_amount];
  265. $user->save();
  266. AccountLogLogic::AccountRecord(
  267. $order['user_id'],
  268. $refund_amount,
  269. 1,
  270. AccountLog::cancel_order_refund,
  271. '',
  272. $order['id'],
  273. $order['order_sn']
  274. );
  275. return true;
  276. }
  277. }