截流自动化的商城平台
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

PayNotifyLogic.php 17KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  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\admin\logic\distribution\DistributionLevelLogic;
  21. use app\api\logic\PayLogic;
  22. use app\common\enum\IntegralGoodsEnum;
  23. use app\common\enum\IntegralOrderEnum;
  24. use app\common\enum\NoticeEnum;
  25. use app\common\model\{AccountLog,
  26. goods\Goods,
  27. integral\IntegralGoods,
  28. integral\IntegralOrder,
  29. order\OrderGoods,
  30. RechargeOrder,
  31. shop\Shop};
  32. use app\common\enum\OrderEnum;
  33. use app\common\enum\OrderLogEnum;
  34. use app\common\enum\PayEnum;
  35. use app\common\model\order\Order;
  36. use app\common\model\order\OrderTrade;
  37. use app\common\server\DistributionServer;
  38. use app\common\model\order\OrderLog;
  39. use app\common\model\user\User;
  40. use app\common\server\ConfigServer;
  41. use think\facade\Db;
  42. use think\Exception;
  43. use think\facade\Log;
  44. /**
  45. * 支付成功后处理订单状态
  46. * Class PayNotifyLogic
  47. * @package app\api\logic
  48. */
  49. class PayNotifyLogic
  50. {
  51. /**
  52. * @notes 回调处理
  53. * @param $action
  54. * @param $order_sn
  55. * @param array $extra
  56. * @return bool|string
  57. * @throws \think\exception\PDOException
  58. * @author suny
  59. * @date 2021/7/13 6:32 下午
  60. */
  61. public static function handle($action, $order_sn, $extra = [])
  62. {
  63. Db::startTrans();
  64. try {
  65. self::$action($order_sn, $extra);
  66. Db::commit();
  67. return true;
  68. } catch (Exception $e) {
  69. Db::rollback();
  70. $record = [
  71. __CLASS__, __FUNCTION__, $e->getFile(), $e->getLine(), $e->getMessage()
  72. ];
  73. Log::write(implode('-', $record));
  74. return $e->getMessage();
  75. }
  76. }
  77. /**
  78. * @notes 添加订单日志表
  79. * @param $order_id
  80. * @param $user_id
  81. * @param $shop_id
  82. * @return array
  83. * @author suny
  84. * @date 2021/7/13 6:32 下午
  85. */
  86. public static function getOrderLogData($order_id, $user_id, $shop_id)
  87. {
  88. $order_log_data = [];
  89. $order_log_data['type'] = OrderLogEnum::TYPE_USER;
  90. $order_log_data['channel'] = OrderLogEnum::USER_PAID_ORDER;
  91. $order_log_data['order_id'] = $order_id;
  92. $order_log_data['handle_id'] = $user_id;
  93. $order_log_data['shop_id'] = $shop_id;
  94. $order_log_data['content'] = OrderLogEnum::getLogDesc(OrderLogEnum::USER_PAID_ORDER);
  95. $order_log_data['create_time'] = time();
  96. return $order_log_data;
  97. }
  98. /**
  99. * @notes 父订单回调
  100. * @param $order_sn
  101. * @param array $extra
  102. * @throws \think\db\exception\DataNotFoundException
  103. * @throws \think\db\exception\DbException
  104. * @throws \think\db\exception\ModelNotFoundException
  105. * @throws \think\exception\DbException
  106. * @author suny
  107. * @date 2021/7/13 6:33 下午
  108. */
  109. private static function trade($order_sn, $extra = [])
  110. {
  111. //根据父订单号查找父订单
  112. $trade = OrderTrade::where(['t_sn' => $order_sn])->find();
  113. //根据父订单id查找子订单
  114. $orders = Order::with(['order_goods', 'shop'])
  115. ->where('trade_id', $trade['id'])
  116. ->select()->toArray();
  117. //修改用户消费累计额度
  118. $user = User::find($trade['user_id']);
  119. $user->total_order_amount = ['inc', $trade['order_amount']];
  120. $user->save();
  121. //赠送成长值
  122. $growth_ratio = ConfigServer::get('transaction', 'money_to_growth', 0);
  123. if ($growth_ratio > 0) {
  124. $able_get_growth = floor($trade['total_amount'] / $growth_ratio);
  125. $user->where('id', $trade['user_id'])
  126. ->inc('user_growth', $able_get_growth)
  127. ->update();
  128. AccountLogLogic::AccountRecord($trade['user_id'], $able_get_growth, 1, AccountLog::order_give_growth, '', $trade['id'], $order_sn);
  129. }
  130. // 生成分销订单
  131. PayLogic::distributionOrderGoods(array_column($orders, 'id'));
  132. // 更新分销会员等级
  133. DistributionLevelLogic::updateDistributionLevel($trade['user_id']);
  134. foreach ($orders as $item) {
  135. //赠送积分
  136. $open_award = ConfigServer::get('order_award', 'open_award', 0);
  137. if ($open_award == 1) {
  138. $award_event = ConfigServer::get('order_award', 'award_event', 0);
  139. $award_ratio = ConfigServer::get('order_award', 'award_ratio', 0);
  140. if ($award_ratio > 0) {
  141. $award_integral = floor($item['order_amount'] * ($award_ratio / 100));
  142. }
  143. }
  144. $orderStatus = Order::STATUS_WAIT_DELIVERY;
  145. //线下自提订单支付完成后订单状态改为待收货
  146. if ($item['delivery_type'] == OrderEnum::DELIVERY_TYPE_SELF) {
  147. $orderStatus = Order::STATUS_WAIT_RECEIVE;
  148. }
  149. //更新订单状态
  150. $data = [
  151. 'pay_status' => PayEnum::ISPAID,
  152. 'pay_time' => time(),
  153. 'order_status' => $orderStatus,
  154. 'award_integral_status' => $award_event ?? 0,
  155. 'award_integral' => $award_integral ?? 0
  156. ];
  157. //如果返回了第三方流水号
  158. if (isset($extra['transaction_id'])) {
  159. $data['transaction_id'] = $extra['transaction_id'];
  160. }
  161. $orderUpdate = Order::update($data, [
  162. [ 'id', '=', $item['id'] ],
  163. [ 'pay_status', '=', PayEnum::UNPAID ],
  164. ]);
  165. if ($orderUpdate->getUpdateResult() <= 0) {
  166. throw new \Exception('修改订单状态失败,订单可能已支付');
  167. }
  168. // 增加一条订单日志
  169. $order_log_add_data = self::getOrderLogData($item['id'], $item['user_id'], $item['shop_id']);
  170. $order_log_datas_insert[] = $order_log_add_data;
  171. OrderLog::insertAll($order_log_datas_insert);
  172. // if ($item['order_type'] == order::NORMAL_ORDER){
  173. // DistributionServer::commission($item['id']);
  174. // }
  175. //通知用户
  176. event('Notice', [
  177. 'scene' => NoticeEnum::ORDER_PAY_NOTICE,
  178. 'mobile' => $item['mobile'] ?? '',
  179. 'params' => ['order_id' => $item['id'], 'user_id' => $item['user_id']]
  180. ]);
  181. //通知商家
  182. if (!empty($item['shop']['mobile'])) {
  183. event('Notice', [
  184. 'scene' => NoticeEnum::USER_PAID_NOTICE_SHOP,
  185. 'mobile' => $item['shop']['mobile'],
  186. 'params' => ['order_id' => $item['id'], 'user_id' => $item['user_id']]
  187. ]);
  188. }
  189. event('Printer', [
  190. 'order_id' => $item['id'],
  191. ]);
  192. }
  193. //如果返回了第三方流水号
  194. if (isset($extra['transaction_id'])) {
  195. $trade->transaction_id = $extra['transaction_id'];
  196. $trade->save();
  197. }
  198. $order_ids = array_column($orders, 'id');
  199. // 虚拟商品更新订单信息
  200. GoodsVirtualLogic::afterPayVirtualDelivery($order_ids);
  201. // 更新商品销量
  202. self::updateGoodsSales($order_ids);
  203. }
  204. /**
  205. * @notes 子订单回调
  206. * @param $order_sn
  207. * @param array $extra
  208. * @throws \think\db\exception\DataNotFoundException
  209. * @throws \think\db\exception\DbException
  210. * @throws \think\db\exception\ModelNotFoundException
  211. * @throws \think\exception\DbException
  212. * @author suny
  213. * @date 2021/7/13 6:33 下午
  214. */
  215. private static function order($order_sn, $extra = [])
  216. {
  217. $time = time();
  218. $order = Order::with(['order_goods', 'shop'])
  219. ->where('order_sn', $order_sn)
  220. ->find()->toArray();
  221. //赠送积分
  222. $open_award = ConfigServer::get('order_award', 'open_award', 0);
  223. if ($open_award == 1) {
  224. $award_event = ConfigServer::get('order_award', 'award_event', 0);
  225. $award_ratio = ConfigServer::get('order_award', 'award_ratio', 0);
  226. if ($award_ratio > 0) {
  227. $award_integral = floor($order['order_amount'] * ($award_ratio / 100));
  228. }
  229. }
  230. $orderStatus = Order::STATUS_WAIT_DELIVERY;
  231. //线下自提订单支付完成后订单状态改为待收货
  232. if ($order['delivery_type'] == OrderEnum::DELIVERY_TYPE_SELF) {
  233. $orderStatus = Order::STATUS_WAIT_RECEIVE;
  234. }
  235. //更新订单状态
  236. $data = [
  237. 'pay_status' => PayEnum::ISPAID,
  238. 'pay_time' => time(),
  239. 'order_status' => $orderStatus,
  240. 'award_integral_status' => $award_event ?? 0,
  241. 'award_integral' => $award_integral ?? 0
  242. ];
  243. //如果返回了第三方流水号
  244. if (isset($extra['transaction_id'])) {
  245. $data['transaction_id'] = $extra['transaction_id'];
  246. }
  247. $orderUpdate = Order::update($data, [
  248. [ 'id', '=', $order['id'] ],
  249. [ 'pay_status', '=', PayEnum::UNPAID ],
  250. ]);
  251. if ($orderUpdate->getUpdateResult() <= 0) {
  252. throw new \Exception('修改订单状态失败,订单可能已支付');
  253. }
  254. // 增加商品销量
  255. // 增加一条订单日志
  256. $order_log_add_data = self::getOrderLogData($order['id'], $order['user_id'], $order['shop_id']);
  257. $order_log_datas_insert[] = $order_log_add_data;
  258. OrderLog::insertAll($order_log_datas_insert);
  259. //修改用户消费累计额度
  260. $user = User::find($order['user_id']);
  261. $user->total_order_amount = ['inc', $order['order_amount']];
  262. $user->save();
  263. //赠送成长值
  264. $growth_ratio = ConfigServer::get('transaction', 'money_to_growth', 0);
  265. if ($growth_ratio > 0) {
  266. $able_get_growth = floor($order['order_amount'] / $growth_ratio);
  267. $user->where('id', $order['user_id'])
  268. ->inc('user_growth', $able_get_growth)
  269. ->update();
  270. AccountLogLogic::AccountRecord($order['user_id'], $able_get_growth, 1, AccountLog::order_give_growth, '', $order['id'], $order_sn);
  271. }
  272. // 生成分销订单
  273. PayLogic::distributionOrderGoods([$order['id']]);
  274. // 更新分销会员等级
  275. DistributionLevelLogic::updateDistributionLevel($order['user_id']);
  276. // //拼购,砍价的订单不参与分销分佣
  277. // if ($order['order_type'] == order::NORMAL_ORDER){
  278. // DistributionServer::commission($order['id']);
  279. // }
  280. // 虚拟商品更新订单信息
  281. GoodsVirtualLogic::afterPayVirtualDelivery($order['id']);
  282. // 更新商品销量
  283. self::updateGoodsSales($order['id']);
  284. //通知用户
  285. event('Notice', [
  286. 'scene' => NoticeEnum::ORDER_PAY_NOTICE,
  287. 'mobile' => $order['mobile'] ?? '',
  288. 'params' => ['order_id' => $order['id'], 'user_id' => $order['user_id']]
  289. ]);
  290. //通知商家
  291. if (!empty($order['shop']['mobile'])) {
  292. event('Notice', [
  293. 'scene' => NoticeEnum::USER_PAID_NOTICE_SHOP,
  294. 'mobile' => $order['shop']['mobile'],
  295. 'params' => ['order_id' => $order['id'], 'user_id' => $order['user_id']]
  296. ]);
  297. }
  298. event('Printer', [
  299. 'order_id' => $order['id'],
  300. ]);
  301. }
  302. /**
  303. * @notes 充值回调
  304. * @param $order_sn
  305. * @param array $extra
  306. * @throws \think\db\exception\DataNotFoundException
  307. * @throws \think\db\exception\DbException
  308. * @throws \think\db\exception\ModelNotFoundException
  309. * @throws \think\exception\DbException
  310. * @author suny
  311. * @date 2021/7/13 6:33 下午
  312. */
  313. private static function recharge($order_sn, $extra = [])
  314. {
  315. $new = time();
  316. $recharge_order = new RechargeOrder();
  317. $order = $recharge_order->where(['order_sn' => $order_sn])->find();
  318. $update_data['pay_time'] = $new;
  319. $update_data['pay_status'] = PayEnum::ISPAID;
  320. if (isset($extra['transaction_id'])) {
  321. $update_data['transaction_id'] = $extra['transaction_id'];
  322. }
  323. $recharge_order->where(['id' => $order['id']])->update($update_data);
  324. $user = User::find($order['user_id']);
  325. $total_money = $order['order_amount'] + $order['give_money'];
  326. $total_integral = $order['give_integral'];
  327. $user->user_money = ['inc', $total_money];
  328. $user->user_integral = ['inc', $total_integral];
  329. $user->user_growth = ['inc', $order['give_growth']];
  330. $user->total_recharge_amount = ['inc', $total_money];
  331. $user->save();
  332. //记录余额
  333. $total_money > 0 && AccountLogLogic::AccountRecord($user->id, $total_money, 1, AccountLog::recharge_money, '', $order['id'], $order_sn);
  334. //记录积分
  335. $total_integral > 0 && AccountLogLogic::AccountRecord($user->id, $total_integral, 1, AccountLog::recharge_give_integral);
  336. //记录成长值
  337. $order['give_growth'] > 0 && AccountLogLogic::AccountRecord($user->id, $order['give_growth'], 1, AccountLog::recharge_give_growth);
  338. }
  339. /**
  340. * @notes 积分订单回调
  341. * @param $order_sn
  342. * @param array $extra
  343. * @throws \think\db\exception\DataNotFoundException
  344. * @throws \think\db\exception\ModelNotFoundException
  345. * @author 段誉
  346. * @date 2022/3/1 14:36
  347. */
  348. private static function integral($order_sn, $extra = [])
  349. {
  350. $order = IntegralOrder::where(['order_sn' => $order_sn])->findOrEmpty();
  351. $goods = $order['goods_snap'];
  352. // 更新订单状态
  353. $data = [
  354. 'order_status' => IntegralOrderEnum::ORDER_STATUS_DELIVERY,
  355. 'pay_status' => PayEnum::ISPAID,
  356. 'pay_time' => time(),
  357. ];
  358. // 红包类型 或者 无需物流 支付完即订单完成
  359. if ($goods['type'] == IntegralGoodsEnum::TYPE_BALANCE || $goods['delivery_way'] == IntegralGoodsEnum::DELIVERY_NO_EXPRESS) {
  360. $data['order_status'] = IntegralOrderEnum::ORDER_STATUS_COMPLETE;
  361. $data['confirm_time'] = time();
  362. }
  363. // 第三方流水号
  364. if (isset($extra['transaction_id'])) {
  365. $data['transaction_id'] = $extra['transaction_id'];
  366. }
  367. IntegralOrder::update($data, ['id' => $order['id']]);
  368. // 更新商品销量
  369. IntegralGoods::where([['id', '=', $goods['id']], ['stock', '>=', $order['total_num']]])
  370. ->dec('stock', $order['total_num'])
  371. ->inc('sales', $order['total_num'])
  372. ->update();
  373. // 红包类型,直接增加余额
  374. if ($goods['type'] == IntegralGoodsEnum::TYPE_BALANCE) {
  375. $reward = round($goods['balance'] * $order['total_num'], 2);
  376. User::where(['id' => $order['user_id']])
  377. ->inc('user_money', $reward)
  378. ->update();
  379. AccountLogLogic::AccountRecord(
  380. $order['user_id'],
  381. $reward, 1,
  382. AccountLog::integral_order_inc_balance,
  383. '', $order['id'], $order['order_sn']
  384. );
  385. }
  386. }
  387. /**
  388. * @notes 更新商品销量
  389. * @param $order_id
  390. * @return bool
  391. * @throws \think\db\exception\DataNotFoundException
  392. * @throws \think\db\exception\DbException
  393. * @throws \think\db\exception\ModelNotFoundException
  394. * @author 段誉
  395. * @date 2022/10/14 10:16
  396. */
  397. private static function updateGoodsSales($order_ids)
  398. {
  399. if (!is_array($order_ids)) {
  400. $order_ids = [$order_ids];
  401. }
  402. $order_goods = OrderGoods::whereIn('order_id', $order_ids)
  403. ->select()
  404. ->toArray();
  405. if (empty($order_goods)) {
  406. return false;
  407. }
  408. foreach ($order_goods as $item) {
  409. // 增加商品销量
  410. Goods::where('id', $item['goods_id'])
  411. ->inc('sales_actual', $item['goods_num'])
  412. ->update();
  413. }
  414. return true;
  415. }
  416. }