截流自动化的商城平台
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.

WechatCorporatePaymentLogic.php 8.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. <?php
  2. namespace app\admin\logic;
  3. /**
  4. * 功能: 微信企业付款
  5. * 用途:企业向微信用户个人付款
  6. * 证书:需要
  7. * 失败后一定要用【原来的商户订单号】去重试,不然有可能存在重复支付的风险!!!
  8. * 商户订单号: 必须唯一且需要为33位以下的数字或字母
  9. */
  10. use EasyWeChat\Factory;
  11. use app\common\model\Pay;
  12. use app\common\server\ConfigServer;
  13. use app\common\server\WeChatServer;
  14. use think\facade\Db;
  15. use app\common\logic\AccountLogLogic;
  16. use app\common\model\AccountLog;
  17. use app\common\model\user\User;
  18. class WechatCorporatePaymentLogic
  19. {
  20. /**
  21. * 获取app
  22. */
  23. public static function getApp($client)
  24. {
  25. $config = WeChatServer::getPayConfigBySource($client)['config'];
  26. $app = Factory::payment($config);
  27. return $app;
  28. }
  29. /**
  30. * 企业付款
  31. */
  32. public static function pay($withdraw)
  33. {
  34. // 微信零钱最小提现金额 1元
  35. if($withdraw['left_money'] < 1) {
  36. return [
  37. 'code' => 0,
  38. 'msg' => '扣除手续费后提现金额不能小于1元'
  39. ];
  40. }
  41. // 每天最多可向同一用户付款10次
  42. $count = Db::name('withdraw_apply')
  43. ->whereTime('update_time', 'd') // 今天
  44. ->where('user_id', $withdraw['user_id'])
  45. ->where('type', 2) // 微信零钱
  46. ->where('status', '>=', 2) // 提现中、提现成功、提现失败
  47. ->count();
  48. if($count > 10) {
  49. return [
  50. 'code' => 0,
  51. 'msg' => '同一个用户一天最多可付款10次'
  52. ];
  53. }
  54. // 一天一个用户累计提现金额不能超过5000元
  55. $sum = Db::name('withdraw_apply')
  56. ->whereTime('update_time', 'd') // 今天
  57. ->where('user_id', $withdraw['user_id'])
  58. ->where('type', 2) // 微信零钱
  59. ->where('status', 'in', [2, 3]) // 提现中、提现成功
  60. ->sum('left_money');
  61. $sum = $sum + $withdraw['left_money'];
  62. if($sum > 5000) {
  63. return [
  64. 'code' => 0,
  65. 'msg' => '一天一个用户累计提现金额(不算手续费)不能超过5000元'
  66. ];
  67. }
  68. // 用户授权信息(同一个用户可能有多条,取client最小的一条)
  69. $userAuth = Db::name('user_auth')->where('user_id', $withdraw['user_id'])->order('client', 'asc')->find();
  70. if(!$userAuth) {
  71. // 无授权记录
  72. return [
  73. 'code'=> 0,
  74. 'msg' => '查询不到该用户的openid'
  75. ];
  76. }
  77. // 获取app
  78. $app = self::getApp($userAuth['client']);
  79. // 商户唯一订单号
  80. $partner_trade_no = $withdraw['sn'];
  81. // 发起企业付款
  82. $result = $app->transfer->toBalance([
  83. // 商户订单号,需保持唯一性(只能是字母或者数字,不能包含有符号)
  84. 'partner_trade_no' => $partner_trade_no,
  85. 'openid' => $userAuth['openid'],
  86. // NO_CHECK:不校验真实姓名, FORCE_CHECK:强校验真实姓名
  87. 'check_name' => 'NO_CHECK',
  88. // 如果 check_name 设置为FORCE_CHECK,则必填用户真实姓名
  89. 're_user_name' => '',
  90. // 企业付款金额,单位为分 100分=1元
  91. 'amount' => 100 * $withdraw['left_money'],
  92. // 企业付款操作说明信息。必填
  93. 'desc' => '佣金提现'
  94. ]);
  95. // 马上将提现申请单状态修改为提现中,并记录微信返回信息,避免同一提现单多次点击发起多次企业付款
  96. $fiterField = ['appid','mch_appid', 'mchid', 'mch_id', 'openid']; // 过滤敏感字段
  97. $filterResult = array_filter($result, function($key) use ($fiterField) {
  98. return !in_array($key, $fiterField);
  99. }, ARRAY_FILTER_USE_KEY);
  100. Db::name('withdraw_apply')
  101. ->where('id', $withdraw['id'])
  102. ->update([
  103. 'status' => 2, // 提现中
  104. 'update_time' => time(),
  105. 'pay_desc' => json_encode($filterResult)
  106. ]);
  107. // 通信标识 return_code
  108. if($result['return_code'] == 'SUCCESS') {
  109. // 业务结果 result_code
  110. if($result['result_code'] == 'SUCCESS') {
  111. // 企业付款成功, 更新提现申请单状态为提现成功并记录支付单号及支付时间
  112. Db::name('withdraw_apply')
  113. ->where('id', $withdraw['id'])
  114. ->update([
  115. 'status' => 3, // 提现成功
  116. 'payment_no' => $result['payment_no'],
  117. 'payment_time' => strtotime($result['payment_time']),
  118. 'update_time' => time()
  119. ]);
  120. return [
  121. 'code' => 1,
  122. 'msg' => '成功提现至微信零钱'
  123. ];
  124. }else{
  125. return [
  126. 'code' => 0,
  127. 'msg' => $result['err_code_des']
  128. ];
  129. }
  130. }else{
  131. // 提现至零钱失败,更新提现申请单为提现失败,并回退佣金
  132. Db::name('withdraw_apply')
  133. ->where('id', $withdraw['id'])
  134. ->update([
  135. 'status' => 4, // 提现失败
  136. 'update_time' => time()
  137. ]);
  138. //回退佣金
  139. $user = User::find($withdraw['user_id']);
  140. $user->earnings = ['inc', $withdraw['money']];
  141. $user->save();
  142. //增加佣金变动记录
  143. AccountLogLogic::AccountRecord(
  144. $withdraw['user_id'],
  145. $withdraw['money'],
  146. 1,
  147. AccountLog::withdraw_back_earnings,
  148. '',
  149. $withdraw['id'],
  150. $withdraw['sn']
  151. );
  152. return [
  153. 'code' => 0,
  154. 'msg' => $result['return_msg']
  155. ];
  156. }
  157. }
  158. /**
  159. * 查询企业付款
  160. */
  161. public static function search($withdraw)
  162. {
  163. $userAuth = Db::name('user_auth')->where('user_id', $withdraw['user_id'])->order('client', 'asc')->find();
  164. if(!$userAuth) {
  165. // 无授权记录
  166. return [
  167. 'code'=> 0,
  168. 'msg' => '查询不到该用户的openid'
  169. ];
  170. }
  171. $app = self::getApp($userAuth['client']);
  172. $partnerTradeNo = $withdraw['sn'];
  173. $result = $app->transfer->queryBalanceOrder($partnerTradeNo);
  174. $fiterField = ['appid','mch_appid', 'mchid', 'mch_id', 'openid']; // 过滤敏感字段
  175. $filterResult = array_filter($result, function($key) use ($fiterField) {
  176. return !in_array($key, $fiterField);
  177. }, ARRAY_FILTER_USE_KEY );
  178. // 记录查询结果
  179. Db::name('withdraw_apply')
  180. ->where('id', $withdraw['id'])
  181. ->update([
  182. 'update_time' => time(),
  183. 'pay_search_desc' => json_encode($filterResult)
  184. ]);
  185. if($result['return_code'] == 'SUCCESS') { // 通信标识
  186. if($result['result_code'] == 'SUCCESS') { // 另一个标识
  187. if($result['status'] == 'SUCCESS') { // 业务结果
  188. // 转账成功,标记提现申请单为提现成功,记录支付信息
  189. Db::name('withdraw_apply')
  190. ->where('id', $withdraw['id'])
  191. ->update([
  192. 'status' => 3, // 提现成功
  193. 'payment_no' => $result['detail_id'],
  194. 'payment_time' => strtotime($result['payment_time']),
  195. 'update_time' => time()
  196. ]);
  197. return [
  198. 'code' => 1,
  199. 'msg' => '已提现成功'
  200. ];
  201. }else if($result['status'] == 'FAILED'){
  202. // 转账失败
  203. Db::name('withdraw_apply')
  204. ->where('id', $withdraw['id'])
  205. ->update([
  206. 'status' => 4, // 提现失败
  207. 'update_time' => time()
  208. ]);
  209. //回退佣金
  210. $user = User::find($withdraw['user_id']);
  211. $user->earnings = ['inc', $withdraw['money']];
  212. $user->save();
  213. //增加佣金变动记录
  214. AccountLogLogic::AccountRecord(
  215. $withdraw['user_id'],
  216. $withdraw['money'],
  217. 1,
  218. AccountLog::withdraw_back_earnings,
  219. '',
  220. $withdraw['id'],
  221. $withdraw['sn']
  222. );
  223. return [
  224. 'code' => 0,
  225. 'msg' => '提现至微信零钱失败'
  226. ];
  227. }else{
  228. return [
  229. 'code' => 0,
  230. 'msg' => $result['reason']
  231. ];
  232. }
  233. }else{
  234. // 查看这个提现单今天是否有重新发起过付款,如果没有则尝试使用【相同的商户订单号】再次发起付款
  235. // $count = Db::name('withdraw_apply')
  236. // ->whereTime('repay_time', 'd') // 今天
  237. // ->where('id', $withdraw['id']) // 提现申请单号
  238. // ->count();
  239. // if(!$count) {
  240. // // 记录重新发起支付的时间
  241. // Db::name('withdraw_apply')->where('id', $withdraw['id'])->update([
  242. // 'repay_time' => time(),
  243. // 'update_time' => time()
  244. // ]);
  245. // return self::pay($withdraw);
  246. // }else{
  247. // return [
  248. // 'code' => 0,
  249. // 'msg' => $result['err_code_des']
  250. // ];
  251. // }
  252. return [
  253. 'code' => 0,
  254. 'msg' => $result['err_code_des']
  255. ];
  256. }
  257. }else{
  258. return [
  259. 'code' => 0,
  260. 'msg' => $result['return_msg']
  261. ];
  262. }
  263. }
  264. }