PayOper.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415
  1. <?php
  2. namespace addons\shopro\service\pay;
  3. use think\Log;
  4. use app\admin\model\shopro\Pay as PayModel;
  5. use app\admin\model\shopro\user\User;
  6. use app\admin\model\shopro\order\Action;
  7. use think\helper\Str;
  8. use addons\shopro\service\Wallet as WalletService;
  9. class PayOper
  10. {
  11. protected $user = null;
  12. /**
  13. * 实例化
  14. *
  15. * @param mixed $user
  16. */
  17. public function __construct($user = null)
  18. {
  19. // 优先使用传入的用户
  20. $this->user = $user ? (is_numeric($user) ? User::get($user) : $user) : auth_user();
  21. }
  22. /**
  23. * 微信预付款
  24. *
  25. * @param think\Model $order
  26. * @param float $money
  27. * @param string $order_type
  28. * @return think\Model
  29. */
  30. public function wechat($order, $money, $order_type = 'order')
  31. {
  32. $pay = $this->addPay($order, [
  33. 'order_type' => $order_type,
  34. 'pay_type' => 'wechat',
  35. 'pay_fee' => $money,
  36. 'real_fee' => $money,
  37. 'transaction_id' => null,
  38. 'payment_json' => [],
  39. 'status' => PayModel::PAY_STATUS_UNPAID
  40. ]);
  41. return $pay;
  42. }
  43. /**
  44. * 支付宝预付款
  45. *
  46. * @param think\Model $order
  47. * @param float $money
  48. * @param string $order_type
  49. * @return think\Model
  50. */
  51. public function alipay($order, $money, $order_type = 'order')
  52. {
  53. $pay = $this->addPay($order, [
  54. 'order_type' => $order_type,
  55. 'pay_type' => 'alipay',
  56. 'pay_fee' => $money,
  57. 'real_fee' => $money,
  58. 'transaction_id' => null,
  59. 'payment_json' => [],
  60. 'status' => PayModel::PAY_STATUS_UNPAID
  61. ]);
  62. return $pay;
  63. }
  64. /**
  65. * 余额付款
  66. *
  67. * @param think\Model $order
  68. * @param float $money
  69. * @param string $order_type
  70. * @return think\Model
  71. */
  72. public function money($order, $money, $order_type = 'order')
  73. {
  74. // 余额支付金额,传入金额和剩余支付金额最大值
  75. $money = $order->remain_pay_fee > $money ? $money : $order->remain_pay_fee; // 混合支付不能超过订单应支付总金额
  76. // 扣除用户余额
  77. WalletService::change($this->user, 'money', -$money, 'order_pay', [
  78. 'order_id' => $order->id,
  79. 'order_sn' => $order->order_sn,
  80. 'order_type' => $order_type,
  81. ]);
  82. // 添加支付记录
  83. $pay = $this->addPay($order, [
  84. 'order_type' => $order_type,
  85. 'pay_type' => 'money',
  86. 'pay_fee' => $money,
  87. 'real_fee' => $money,
  88. 'transaction_id' => null,
  89. 'payment_json' => [],
  90. 'status' => PayModel::PAY_STATUS_PAID
  91. ]);
  92. // 余额直接支付成功,更新订单剩余应付款金额,并检测订单状态
  93. return $this->checkAndPaid($order, $order_type);
  94. }
  95. /**
  96. * 积分支付
  97. *
  98. * @param think\Model $order
  99. * @param float $money
  100. * @param string $order_type
  101. * @return think\Model
  102. */
  103. public function score($order, $score, $order_type = 'order')
  104. {
  105. if ($order_type == 'order') {
  106. if ($order['type'] == 'score') {
  107. $log_type = 'score_shop_pay';
  108. $real_fee = $score; // 积分商城真实抵扣,就是积分
  109. } else {
  110. $log_type = 'order_pay';
  111. // $real_fee = ; // 积分商城真实抵扣,就是积分
  112. error_stop('缺少积分抵扣金额'); // 支持积分抵扣时补全
  113. }
  114. }
  115. WalletService::change($this->user, 'score', -$score, $log_type, [
  116. 'order_id' => $order->id,
  117. 'order_sn' => $order->order_sn,
  118. 'order_type' => $order_type,
  119. ]);
  120. // 添加支付记录
  121. $pay = $this->addPay($order, [
  122. 'order_type' => $order_type,
  123. 'pay_type' => 'score',
  124. 'pay_fee' => $score,
  125. 'real_fee' => $real_fee,
  126. 'transaction_id' => null,
  127. 'payment_json' => [],
  128. 'status' => PayModel::PAY_STATUS_PAID
  129. ]);
  130. // 积分直接支付成功,更新订单剩余应付款金额,并检测订单状态
  131. return $this->checkAndPaid($order, $order_type);
  132. }
  133. /**
  134. * 线下支付(货到付款)
  135. *
  136. * @param think\Model $order
  137. * @param float $money
  138. * @param string $order_type
  139. * @return think\Model
  140. */
  141. public function offline($order, $money, $order_type = 'order')
  142. {
  143. // 添加支付记录
  144. $pay = $this->addPay($order, [
  145. 'order_type' => $order_type,
  146. 'pay_type' => 'offline',
  147. 'pay_fee' => $money,
  148. 'real_fee' => $money,
  149. 'transaction_id' => null,
  150. 'payment_json' => [],
  151. 'status' => PayModel::PAY_STATUS_PAID
  152. ]);
  153. // 更新订单剩余应付款金额,并检测订单状态
  154. return $this->checkAndPaid($order, $order_type, 'offline');
  155. }
  156. /**
  157. * 微信支付宝支付回调通用方法
  158. *
  159. * @param \think\Model $pay
  160. * @param array $notify
  161. * @return void
  162. */
  163. public function notify($pay, $notify)
  164. {
  165. $pay->status = PayModel::PAY_STATUS_PAID;
  166. $pay->transaction_id = $notify['transaction_id'];
  167. $pay->buyer_info = $notify['buyer_info'];
  168. $pay->payment_json = $notify['payment_json'];
  169. $pay->paid_time = time();
  170. $pay->save();
  171. $orderModel = $this->getOrderModel($pay->order_type);
  172. $order = new $orderModel();
  173. $order = $order->where('id', $pay->order_id)->find();
  174. if (!$order) {
  175. // 订单未找到,非正常情况,这里记录日志
  176. Log::write('pay-notify-error:order notfound;pay:' . json_encode($pay) . ';notify:' . json_encode($notify));
  177. return false;
  178. }
  179. if ($order->status == $order::STATUS_UNPAID) { // 未支付,检测支付状态
  180. $order = $this->checkAndPaid($order, $pay->order_type);
  181. }
  182. return $order;
  183. }
  184. /**
  185. * 更新订单剩余应支付金额,并且检测订单状态
  186. *
  187. * @param think\Model $order
  188. * @param string $order_type
  189. * @return think\Model
  190. */
  191. public function checkAndPaid($order, $order_type, $pay_mode = 'online')
  192. {
  193. // 获取订单已支付金额
  194. $payed_fee = $this->getPayedFee($order, $order_type);
  195. $remain_pay_fee = bcsub($order->pay_fee, (string)$payed_fee, 2);
  196. $order->remain_pay_fee = $remain_pay_fee;
  197. if ($remain_pay_fee <= 0) {
  198. $order->remain_pay_fee = 0;
  199. $order->paid_time = time();
  200. $order->status = $order::STATUS_PAID;
  201. } else {
  202. if ($pay_mode == 'offline') {
  203. // 订单未支付成功,并且是线下支付(货到付款),将订单状态改为 pending
  204. $order->status = $order::STATUS_PENDING;
  205. $order->ext = array_merge($order->ext, ['pending_time' => time()]); // 货到付款下单时间
  206. $order->pay_mode = 'offline';
  207. }
  208. }
  209. $order->save();
  210. if ($order->status == $order::STATUS_PAID) {
  211. // 订单支付完成
  212. $user = User::where('id', $order->user_id)->find();
  213. if ($order_type == 'order') {
  214. if ($pay_mode == 'offline') {
  215. Action::add($order, null, auth_admin(), 'admin', '管理员操作自动货到付款支付成功');
  216. // 在控制器执行后续内容,这里不再处理
  217. return $order;
  218. } else {
  219. Action::add($order, null, $user, 'user', '用户支付成功');
  220. // 支付成功后续使用异步队列处理
  221. \think\Queue::push('\addons\shopro\job\OrderPaid@paid', ['order' => $order, 'user' => $user], 'shopro-high');
  222. }
  223. } else if ($order_type == 'trade_order') {
  224. // 支付成功后续使用异步队列处理
  225. \think\Queue::push('\addons\shopro\job\trade\OrderPaid@paid', ['order' => $order, 'user' => $user], 'shopro-high');
  226. }
  227. } else if ($order->status == $order::STATUS_PENDING) {
  228. // 货到付款,添加货到付款队列(后续也需要处理拼团, 减库存等等)
  229. $user = User::where('id', $order->user_id)->find();
  230. if ($order_type == 'order') {
  231. Action::add($order, null, $user, 'user', '用户货到付款下单成功');
  232. // 支付成功后续使用异步队列处理
  233. \think\Queue::push('\addons\shopro\job\OrderPaid@offline', ['order' => $order, 'user' => $user], 'shopro-high');
  234. }
  235. }
  236. return $order;
  237. }
  238. /**
  239. * 获取订单已支付金额,商城订单 计算 积分抵扣金额
  240. *
  241. * @param \think\Model $order
  242. * @param string $order_type
  243. * @return string
  244. */
  245. public function getPayedFee($order, $order_type)
  246. {
  247. // 锁定读取所有已支付的记录,判断已支付金额
  248. $pays = PayModel::{'type' . Str::studly($order_type)}()->paid()->where('order_id', $order->id)->lock(true)->select();
  249. // 商城或者积分商城订单
  250. $payed_fee = '0';
  251. foreach ($pays as $key => $pay) {
  252. if ($pay->pay_type == 'score') {
  253. if ($order_type == 'order' && $order['type'] == 'goods') {
  254. // 商城类型订单,并且不是积分商城订单,加上积分抵扣真实金额
  255. $payed_fee = bcadd($payed_fee, $pay->real_fee, 2);
  256. } else {
  257. // 其他类型,需要计算积分抵扣的金额时
  258. }
  259. } else {
  260. $payed_fee = bcadd($payed_fee, $pay->real_fee, 2);
  261. }
  262. }
  263. return $payed_fee;
  264. }
  265. /**
  266. * 获取剩余可退款的pays 记录(不含积分抵扣)
  267. *
  268. * @param integer $order_id
  269. * @param string $sort 排序:money=优先退回余额支付的钱
  270. * @return \think\Collection
  271. */
  272. public function getCanRefundPays($order_id, $sort = 'money')
  273. {
  274. // 商城订单,已支付的 pay 记录, 这里只查 钱的支付记录,不查积分
  275. $pays = PayModel::typeOrder()->paid()->isMoney()->where('order_id', $order_id)->lock(true)->order('id', 'asc')->select();
  276. $pays = collection($pays);
  277. if ($sort == 'money') {
  278. // 对 pays 进行排序,优先退 money 的钱
  279. $pays = $pays->sort(function ($a, $b) {
  280. if ($a['pay_type'] == 'money' && $b['pay_type'] == 'money') {
  281. return 0;
  282. } else if ($a['pay_type'] == 'money' && $b['pay_type'] != 'money') {
  283. return -1;
  284. } else if ($a['pay_type'] != 'money' && $b['pay_type'] == 'money') {
  285. return 1;
  286. } else {
  287. return 0;
  288. }
  289. });
  290. $pays = $pays->values();
  291. }
  292. return $pays;
  293. }
  294. /**
  295. * 获取剩余可退款金额,不含积分相关支付
  296. *
  297. * @param mixed $pays
  298. * @return string
  299. */
  300. public function getRemainRefundMoney($pays)
  301. {
  302. // 拿到 所有可退款的支付记录
  303. $pays = ($pays instanceof \think\Collection) ? $pays : $this->getCanRefundPays($pays);
  304. // 支付金额,除了已经退完款的金额 (这里不退积分)
  305. $payed_money = (string)array_sum($pays->column('pay_fee'));
  306. // 已经退款金额 (这里不退积分)
  307. $refunded_money = (string)array_sum($pays->column('refund_fee'));
  308. // 当前剩余的最大可退款金额,支付金额 - 已退款金额
  309. $remain_max_refund_money = bcsub($payed_money, $refunded_money, 2);
  310. return $remain_max_refund_money;
  311. }
  312. /**
  313. * 添加 pay 记录
  314. *
  315. * @param think\Model $order
  316. * @param array $params
  317. * @return think\Model
  318. */
  319. public function addPay($order, $params)
  320. {
  321. $payModel = new PayModel();
  322. $payModel->order_type = $params['order_type'];
  323. $payModel->order_id = $order->id;
  324. $payModel->pay_sn = get_sn($this->user->id, 'P');
  325. $payModel->user_id = $this->user->id;
  326. $payModel->pay_type = $params['pay_type'];
  327. $payModel->pay_fee = $params['pay_fee'];
  328. $payModel->real_fee = $params['real_fee'];
  329. $payModel->transaction_id = $params['transaction_id'];
  330. $payModel->payment_json = $params['payment_json'];
  331. $payModel->paid_time = $params['status'] == PayModel::PAY_STATUS_PAID ? time() : null;
  332. $payModel->status = $params['status'];
  333. $payModel->refund_fee = 0;
  334. $payModel->save();
  335. return $payModel;
  336. }
  337. public function getOrderModel($order_type)
  338. {
  339. switch ($order_type) {
  340. case 'trade_order':
  341. $orderModel = '\\app\\admin\\model\\shopro\\trade\\Order';
  342. break;
  343. case 'order':
  344. $orderModel = '\\app\\admin\\model\\shopro\\order\\Order';
  345. break;
  346. default:
  347. $orderModel = '\\app\\admin\\model\\shopro\\order\\Order';
  348. break;
  349. }
  350. return $orderModel;
  351. }
  352. }