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

OrderRefundLogic.php 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  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\OrderEnum;
  21. use app\common\enum\OrderGoodsEnum;
  22. use app\common\enum\OrderLogEnum;
  23. use app\common\enum\OrderRefundEnum;
  24. use app\common\enum\PayEnum;
  25. use app\common\model\order\Order;
  26. use app\common\model\AccountLog;
  27. use app\common\model\order\Order as CommonOrder;
  28. use app\common\model\order\OrderGoods;
  29. use app\common\model\order\OrderLog;
  30. use app\common\model\Pay;
  31. use app\common\model\user\User;
  32. use app\common\server\AliPayServer;
  33. use app\common\server\DouGong\pay\PayZhengsaoRefund;
  34. use app\common\server\WeChatPayServer;
  35. use app\common\server\WeChatServer;
  36. use think\Exception;
  37. use think\facade\Db;
  38. use think\facade\Event;
  39. use think\facade\Log;
  40. /**
  41. * 订单退款逻辑
  42. * Class OrderRefundLogic
  43. * @package app\common\logic
  44. */
  45. class OrderRefundLogic
  46. {
  47. /**
  48. * Notes: 取消订单
  49. * @param $order_id
  50. * @param int $handle_type
  51. * @param int $handle_id
  52. * @author 段誉(2021/1/28 15:23)
  53. * @return Order
  54. */
  55. public static function cancelOrder($order_id, $handle_type = OrderLogEnum::TYPE_SYSTEM, $handle_id = 0)
  56. {
  57. //更新订单状态
  58. // $order = order::get($order_id);
  59. $order = Order::where('id',$order_id)->find();
  60. $order->order_status = OrderEnum::ORDER_STATUS_DOWN;
  61. $order->update_time = time();
  62. $order->cancel_time = time();
  63. $order->save();
  64. // 取消订单后的操作
  65. switch ($handle_type) {
  66. case OrderLogEnum::TYPE_USER:
  67. $channel = OrderLogEnum::USER_CANCEL_ORDER;
  68. break;
  69. case OrderLogEnum::TYPE_SHOP:
  70. $channel = OrderLogEnum::SHOP_CANCEL_ORDER;
  71. break;
  72. case OrderLogEnum::TYPE_SYSTEM:
  73. $channel = OrderLogEnum::SYSTEM_CANCEL_ORDER;
  74. break;
  75. }
  76. event('AfterCancelOrder', [
  77. 'type' => $handle_type,
  78. 'channel' => $channel,
  79. 'order_id' => $order->id,
  80. 'handle_id' => $handle_id,
  81. ]);
  82. }
  83. /**
  84. * @notes 处理订单退款(事务在取消订单逻辑处)
  85. * @param $order
  86. * @param $order_amount
  87. * @param $refund_amount
  88. * @return bool
  89. * @throws Exception
  90. * @throws \think\db\exception\DataNotFoundException
  91. * @throws \think\db\exception\ModelNotFoundException
  92. * @author 段誉
  93. * @date 2021/12/20 15:13
  94. */
  95. public static function refund($order, $order_amount, $refund_amount,$refund_way = OrderRefundEnum::REFUND_WAY_ORIGINAL)
  96. {
  97. //退款记录
  98. $refund_id = self::addRefundLog($order, $order_amount, $refund_amount);
  99. if ($refund_amount <= 0) {
  100. return false;
  101. }
  102. $pay_way = $order['pay_way'];
  103. if ($refund_way == OrderRefundEnum::REFUND_WAY_BALANCE) {
  104. $pay_way = PayEnum::BALANCE_PAY;
  105. }
  106. switch ($pay_way) {
  107. //余额退款
  108. case PayEnum::BALANCE_PAY:
  109. self::balancePayRefund($order, $refund_amount);
  110. break;
  111. //微信退款
  112. case PayEnum::WECHAT_PAY:
  113. self::wechatPayRefund($order, $refund_id);
  114. break;
  115. //支付宝退款
  116. case PayEnum::ALI_PAY:
  117. self::aliPayRefund($order, $refund_id);
  118. break;
  119. case PayEnum::HFDG_WECHAT:
  120. case PayEnum::HFDG_ALIPAY:
  121. $payZsRefund = new PayZhengsaoRefund([
  122. 'refund' => [
  123. 'id' => $refund_id,
  124. 'money' => $refund_amount,
  125. ],
  126. 'order' => [
  127. 'id' => $order['id'],
  128. 'transaction_id' => $order['transaction_id'],
  129. 'hfdg_params' => $order['hfdg_params'],
  130. ],
  131. 'from' => 'order',
  132. ]);
  133. $result = $payZsRefund->request()->getRefundResult();
  134. if ($result['code'] != 1) {
  135. throw new \Exception($result['msg']);
  136. }
  137. break;
  138. }
  139. return true;
  140. }
  141. /**
  142. * Notes: 微信支付退款
  143. * @param $order mixed (订单信息)
  144. * @param $refund_id mixed (退款记录id) 
  145. * @author 段誉(2021/1/27 16:04)
  146. * @throws Exception
  147. * @throws \think\db\exception\DataNotFoundException
  148. * @throws \think\db\exception\ModelNotFoundException
  149. * @throws \think\exception\DbException
  150. * @throws \think\exception\PDOException
  151. */
  152. public static function wechatPayRefund($order, $refund_id)
  153. {
  154. $config = WeChatServer::getPayConfigBySource($order['order_source'])['config'];
  155. if (empty($config)) {
  156. throw new Exception('请联系管理员设置微信相关配置!');
  157. }
  158. if (!isset($config['cert_path']) || !isset($config['key_path'])) {
  159. throw new Exception('请联系管理员设置微信证书!');
  160. }
  161. if (!file_exists($config['cert_path']) || !file_exists($config['key_path'])) {
  162. throw new Exception('微信证书不存在,请联系管理员!');
  163. }
  164. $refund_log = Db::name('order_refund')->where(['id' => $refund_id])->find();
  165. $total_fee = Db::name('order_trade')->where(['transaction_id' => $order['transaction_id']])->value('order_amount');
  166. // 单独支付的子订单 父订单未记录transaction_id 使用子订单的金额
  167. $total_fee = $total_fee ? : $order['order_amount'];
  168. $data = [
  169. 'transaction_id' => $order['transaction_id'],
  170. 'refund_sn' => $refund_log['refund_sn'],
  171. 'total_fee' => bcmul($total_fee, 100),//订单金额,单位为分
  172. 'refund_fee' => bcmul($refund_log['refund_amount'], 100),//退款金额
  173. ];
  174. $result = WeChatPayServer::refund($config, $data);
  175. if (isset($result['return_code']) && $result['return_code'] == 'FAIL') {
  176. throw new Exception($result['return_msg']);
  177. }
  178. if (isset($result['err_code_des'])) {
  179. throw new Exception($result['err_code_des']);
  180. }
  181. if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
  182. $update_data = [
  183. 'wechat_refund_id' => $result['refund_id'] ?? 0,
  184. 'refund_msg' => json_encode($result, JSON_UNESCAPED_UNICODE),
  185. ];
  186. //更新退款日志记录
  187. Db::name('order_refund')->where(['id' => $refund_id])->update($update_data);
  188. } else {
  189. throw new Exception('微信支付退款失败');
  190. }
  191. }
  192. /**
  193. * Notes: 支付宝退款
  194. * @param $order
  195. * @param $refund_id
  196. * @author 段誉(2021/3/23 15:48)
  197. * @throws Exception
  198. * @throws \think\exception\PDOException
  199. */
  200. public static function aliPayRefund($order, $refund_id)
  201. {
  202. $refund_log = Db::name('order_refund')->where(['id' => $refund_id])->find();
  203. $trade_id = $order['trade_id'];
  204. $trade = Db::name('order_trade')->where(['id' => $trade_id])->find();
  205. $result = (new AliPayServer())->refund($trade['t_sn'], $refund_log['refund_amount'], $refund_log['refund_sn']);
  206. // $result = (array)$result
  207. if ($result['code'] == '10000' && $result['msg'] == 'Success' && $result['fund_change'] == 'Y') {
  208. //更新退款日志记录
  209. $update_data = [ 'refund_msg' => json_encode($result['msg'], JSON_UNESCAPED_UNICODE) ];
  210. Db::name('order_refund')->where(['id' => $refund_id])->update($update_data);
  211. } else {
  212. $result = (new AliPayServer())->refund($order['order_sn'], $refund_log['refund_amount'], $refund_log['refund_sn']);
  213. // $result = (array)$result;
  214. if ($result['code'] == '10000' && $result['msg'] == 'Success' && $result['fund_change'] == 'Y') {
  215. //更新退款日志记录
  216. $update_data = [ 'refund_msg' => json_encode($result['msg'], JSON_UNESCAPED_UNICODE) ];
  217. Db::name('order_refund')->where(['id' => $refund_id])->update($update_data);
  218. }else{
  219. throw new Exception('支付宝退款失败');
  220. }
  221. }
  222. }
  223. /**
  224. * Notes: 增加退款记录
  225. * @param $order
  226. * @param $order_amount
  227. * @param $refund_amount
  228. * @param string $result_msg
  229. * @author 段誉(2021/1/28 15:23)
  230. * @return int|string
  231. * @throws \think\db\exception\DataNotFoundException
  232. * @throws \think\db\exception\ModelNotFoundException
  233. * @throws \think\exception\DbException
  234. */
  235. public static function addRefundLog($order, $order_amount, $refund_amount, $result_msg = '退款成功')
  236. {
  237. $data = [
  238. 'order_id' => $order['id'],
  239. 'user_id' => $order['user_id'],
  240. 'refund_sn' => createSn('order_refund', 'refund_sn'),
  241. 'order_amount' => $order_amount,
  242. 'refund_amount' => $refund_amount,
  243. 'transaction_id' => $order['transaction_id'],
  244. 'create_time' => time(),
  245. 'refund_status' => 1,
  246. 'refund_at' => time(),
  247. 'refund_msg' => json_encode($result_msg, JSON_UNESCAPED_UNICODE),
  248. ];
  249. return Db::name('order_refund')->insertGetId($data);
  250. }
  251. /**
  252. * Notes: 取消订单,退款后更新订单和订单商品信息
  253. * @param $order
  254. * @author 段誉(2021/1/28 14:21)
  255. * @throws Exception
  256. * @throws \think\exception\PDOException
  257. */
  258. public static function cancelOrderRefundUpdate($order)
  259. {
  260. //订单商品=>标记退款成功状态
  261. Db::name('order_goods')
  262. ->where(['order_id' => $order['id']])
  263. ->update(['refund_status' => OrderGoodsEnum::REFUND_STATUS_SUCCESS]);
  264. //更新订单支付状态为已退款
  265. Db::name('order')->where(['id' => $order['id']])->update([
  266. 'pay_status' => PayEnum::REFUNDED,
  267. 'refund_status' => OrderEnum::REFUND_STATUS_ALL_REFUND,//订单退款状态; 0-未退款;1-部分退款;2-全部退款
  268. 'refund_amount' => $order['order_amount'],
  269. 'is_cancel' => OrderEnum::ORDER_CANCEL,
  270. ]);
  271. }
  272. /**
  273. * Notes:售后退款更新订单或订单商品状态
  274. * @param $order
  275. * @param $order_goods_id
  276. * @author 段誉(2021/1/28 15:22)
  277. */
  278. public static function afterSaleRefundUpdate($order, $order_goods_id, $admin_id = 0)
  279. {
  280. $order_goods = OrderGoods::find(['id' => $order_goods_id]);
  281. $order_goods->refund_status = OrderGoodsEnum::REFUND_STATUS_SUCCESS;//退款成功
  282. $order_goods->save();
  283. //更新订单状态
  284. $order = Order::find(['id' => $order['id']]);
  285. $order->pay_status = PayEnum::REFUNDED;
  286. $order->refund_amount += $order_goods['total_pay_price'];//退款金额 + 以前的退款金额
  287. $order->refund_status = 1;//退款状态:0-未退款;1-部分退款;2-全部退款
  288. //如果订单商品已全部退款
  289. if (self::checkOrderGoods($order['id'])) {
  290. $order->order_status = CommonOrder::STATUS_CLOSE;
  291. $order->refund_status = 2;
  292. OrderLogLogic::record(
  293. OrderLogEnum::TYPE_SHOP,
  294. OrderLogEnum::SYSTEM_CANCEL_ORDER,
  295. $order['id'],
  296. $admin_id,
  297. OrderLogEnum::getLogDesc(OrderLogEnum::SYSTEM_CANCEL_ORDER)
  298. );
  299. }
  300. $order->save();
  301. }
  302. //订单内商品是否已全部
  303. public static function checkOrderGoods($order_id)
  304. {
  305. $order_goods = OrderGoods::where('order_id', $order_id)->select();
  306. if (empty($order_goods)) {
  307. return false;
  308. }
  309. foreach ($order_goods as $item) {
  310. if ($item['refund_status'] != OrderGoodsEnum::REFUND_STATUS_SUCCESS) {
  311. return false;
  312. }
  313. }
  314. return true;
  315. }
  316. /**
  317. * Notes: 余额退款
  318. * @param $order
  319. * @param $refund_amount
  320. * @author 段誉(2021/1/28 15:24)
  321. * @return bool
  322. * @throws \think\db\exception\DataNotFoundException
  323. * @throws \think\db\exception\ModelNotFoundException
  324. * @throws \think\exception\DbException
  325. */
  326. public static function balancePayRefund($order, $refund_amount)
  327. {
  328. $user = User::find($order['user_id']);
  329. $user->user_money = ['inc', $refund_amount];
  330. $user->save();
  331. AccountLogLogic::AccountRecord(
  332. $order['user_id'],
  333. $refund_amount,
  334. 1,
  335. AccountLog::cancel_order_refund,
  336. '',
  337. $order['id'],
  338. $order['order_sn']
  339. );
  340. return true;
  341. }
  342. /**
  343. * Notes: 退款失败增加错误记录
  344. * @param $order
  345. * @param $err_msg
  346. * @author 段誉(2021/1/28 15:24)
  347. * @return int|string
  348. * @throws \think\db\exception\DataNotFoundException
  349. * @throws \think\db\exception\ModelNotFoundException
  350. * @throws \think\exception\DbException
  351. */
  352. public static function addErrorRefund($order, $err_msg)
  353. {
  354. $refund_data = [
  355. 'order_id' => $order['id'],
  356. 'user_id' => $order['user_id'],
  357. 'refund_sn' => createSn('order_refund', 'refund_sn'),
  358. 'order_amount' => $order['order_amount'],//订单应付金额
  359. 'refund_amount' => $order['order_amount'],//订单退款金额
  360. 'transaction_id' => $order['transaction_id'],
  361. 'create_time' => time(),
  362. 'refund_status' => 2,
  363. 'refund_msg' => json_encode($err_msg, JSON_UNESCAPED_UNICODE),
  364. ];
  365. return Db::name('order_refund')->insertGetId($refund_data);
  366. }
  367. }