WechatNewPay.php 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704
  1. <?php
  2. /**
  3. * 重庆赤晓店信息科技有限公司
  4. * https://www.chixiaodian.com
  5. * Copyright (c) 2023 赤店商城 All rights reserved.
  6. */
  7. namespace app\utils\Wechat;
  8. use app\models\LevelOrder;
  9. use app\models\Option;
  10. use app\models\Order;
  11. use app\models\OrderUnion;
  12. use app\models\ReOrder;
  13. use app\models\SaasUser;
  14. use app\models\Store;
  15. use app\models\WechatConfig;
  16. use app\modules\client\models\v1\OpenidForm;
  17. use app\utils\DataTransform;
  18. use app\utils\OrderNo;
  19. use EasyWeChat\Kernel\Exceptions\Exception;
  20. use GuzzleHttp\Exception\GuzzleException;
  21. use yii\base\BaseObject;
  22. use function GuzzleHttp\Psr7\str;
  23. use app\models\DeliveryInfo;
  24. use app\models\AccountLog;
  25. /**
  26. * 微信支付统一入口
  27. * Class WechatPay
  28. * @package app\utils\Wechat
  29. */
  30. class WechatNewPay extends WechatProfit
  31. {
  32. // TODO: 微信走统一的订单回调接口
  33. public static $notify_url = 'wechat/notify';
  34. /**
  35. * 下单
  36. * @param string $goods_names
  37. * @param array | Order | \app\plugins\scanCodePay\models\Order |ReOrder $order | PtOrder | YyOrder | LevelOrder $order
  38. * @param integer $type
  39. * @param integer $total_pay_price
  40. * @param boolean $is_app
  41. * @param boolean $is_h5
  42. * @return array
  43. */
  44. public static function orderUnify($order, $type, $goods_names = null, $total_pay_price = 0, $is_app = false, $balance_price = 0, $is_h5 = false, $is_official = false, $open_id = '')
  45. {
  46. if ($is_app) {
  47. $from_type = WechatConfig::TYPE_CONFIG_APP;
  48. } else if ($is_official) {
  49. $from_type = WechatConfig::TYPE_CONFIG_MP;
  50. } else if ($is_h5) {
  51. $from_type = WechatConfig::TYPE_CONFIG_H5;
  52. } else {
  53. $from_type = WechatConfig::TYPE_CONFIG_MINI;
  54. }
  55. $is_platform = empty($order->is_platform) ? 0 : 1;
  56. try {
  57. parent::init(get_store_id(), $is_platform, 0, $from_type);
  58. } catch (\Exception $e) {
  59. return [
  60. 'code' => 1,
  61. 'msg' => $e->getMessage()
  62. ];
  63. }
  64. if (!$order || !in_array($type, OrderNo::$validOrderType) || ($type != OrderNo::ORDER_UNION && !$goods_names)) {
  65. return [
  66. 'code' => 1,
  67. 'msg' => '订单信息或订单类型错误'
  68. ];
  69. }
  70. $store_wechat = WechatConfig::findOne(['store_id' => get_store_id(), 'type' => $from_type]);
  71. $mch_id = $store_wechat->mch_id;
  72. if (!$mch_id) {
  73. $merchant_info = \app\models\MerchantInfo::findOne(['is_use' => 1, 'bind_store_id' => get_store_id(), 'status' => 3, 'is_delete' => 0]);
  74. if ($merchant_info->sub_mch_id) {
  75. $mch_id = $merchant_info->sub_mch_id;
  76. }
  77. }
  78. if ($is_official) {
  79. $order->order_origin = Order::ORDER_SOURCE_MP;
  80. $order->save();
  81. }
  82. if (!$open_id) {
  83. if ($is_app) {
  84. $open_id = get_user()->wechat_app_open_id;
  85. } else if ($is_official) {
  86. $open_id = get_user()->wechat_platform_open_id;
  87. } else {
  88. $open_id = get_user()->wechat_open_id;
  89. }
  90. }
  91. if ($type == OrderNo::ORDER_UNION) {
  92. $config = [
  93. 'body' => count($order) . '笔订单合并支付',
  94. 'out_trade_no' => OrderNo::getOrderNo(OrderNo::ORDER_UNION),
  95. 'total_fee' => $balance_price > 0 ? floatval($total_pay_price - $balance_price) * 100 : $total_pay_price * 100,
  96. 'notify_url' => pay_notify_url(self::$notify_url),
  97. 'sub_appid' => $store_wechat->app_id,
  98. 'sub_mch_id' => $mch_id,
  99. 'sub_openid' => $open_id,
  100. ];
  101. if (is_platform()) {
  102. $config['sub_appid'] = Option::get('platform_appid', 0, 'saas')['value'];
  103. $config['sub_openid'] = SaasUser::findOne(['mobile' => get_user()->binding])->platform_open_id;
  104. }
  105. $self_mini = Option::get('self_mini', get_store_id(), 'store', 0)['value'];
  106. if (\Yii::$app->prod_is_dandianpu() && !$self_mini) {
  107. if ($from_type == WechatConfig::TYPE_CONFIG_MINI) {
  108. $config['sub_appid'] = Option::get('one_store_wechat_appid', 0, 'saas', '')['value'];
  109. }
  110. if ($from_type == WechatConfig::TYPE_CONFIG_MP) {
  111. $config['sub_appid'] = Option::get('one_store_wechat_official_appid', 0, 'saas', '')['value'];
  112. }
  113. if ($from_type == WechatConfig::TYPE_CONFIG_H5) {
  114. unset($config['sub_appid']);
  115. }
  116. }
  117. if ($is_h5) {
  118. $config['trade_type'] = 'MWEB';
  119. } else {
  120. $config['trade_type'] = 'JSAPI';
  121. }
  122. $order_union = new OrderUnion();
  123. $order_union->store_id = get_store_id();
  124. $order_union->user_id = get_user()->id;
  125. $order_union->order_no = $config['out_trade_no'];
  126. $order_union->price = $total_pay_price;
  127. $order_union->is_pay = 0;
  128. $order_union->created_at = time();
  129. $order_union->is_delete = 0;
  130. $order_id_list = [];
  131. foreach ($order as $value) {
  132. $order_id_list[] = $value->id;
  133. }
  134. $order_union->order_id_list = json_encode($order_id_list);
  135. if (!$order_union->save()) {
  136. foreach ($order_union->errors as $error) {
  137. return [
  138. 'code' => 1,
  139. 'msg' => $error
  140. ];
  141. }
  142. }
  143. } else {
  144. $pay_price = $order->pay_price;
  145. // if ($type != OrderNo::ORDER_RECHARGE) {
  146. // if (isset($order->is_delivery) && $order->is_delivery == 1) {
  147. // $deliveryInfo = DeliveryInfo::find()->where(['order_no' => $order->order_no])->one();
  148. // if ($deliveryInfo) {
  149. // $pay_price += $deliveryInfo->fee;
  150. // }
  151. // }
  152. // }
  153. $config = [
  154. 'body' => $goods_names,
  155. 'out_trade_no' => $order->order_no,
  156. 'total_fee' => $balance_price > 0 ? floatval($pay_price - $balance_price) * 100 : $pay_price * 100,
  157. 'notify_url' => pay_notify_url(self::$notify_url),
  158. 'sub_appid' => $store_wechat->app_id,
  159. 'sub_mch_id' => $mch_id,
  160. 'sub_openid' => $open_id,
  161. ];
  162. if (is_platform()) {
  163. $config['sub_appid'] = Option::get('platform_appid', 0, 'saas')['value'];
  164. $config['sub_openid'] = SaasUser::findOne(['mobile' => get_user()->binding])->platform_open_id;
  165. }
  166. $self_mini = Option::get('self_mini', get_store_id(), 'store', 0)['value'];
  167. if (\Yii::$app->prod_is_dandianpu() && !$self_mini) {
  168. if ($from_type == WechatConfig::TYPE_CONFIG_MINI) {
  169. $config['sub_appid'] = Option::get('one_store_wechat_appid', 0, 'saas', '')['value'];
  170. $config['sub_openid'] = get_user()->wechat_open_id;
  171. }
  172. if ($from_type == WechatConfig::TYPE_CONFIG_MP) {
  173. $config['sub_appid'] = Option::get('one_store_wechat_official_appid', 0, 'saas', '')['value'];
  174. $config['sub_openid'] = get_user()->wechat_platform_open_id;
  175. }
  176. if ($from_type == WechatConfig::TYPE_CONFIG_H5) {
  177. unset($config['sub_appid']);
  178. }
  179. }
  180. if ($is_h5) {
  181. $config['trade_type'] = 'MWEB';
  182. } else {
  183. $config['trade_type'] = 'JSAPI';
  184. }
  185. if ($is_app) {
  186. $config['trade_type'] = 'APP';
  187. }
  188. }
  189. // 是否走分账
  190. if (is_profit_sharing()) {
  191. $config['profit_sharing'] = 'Y';
  192. }
  193. \Yii::error($config);
  194. try {
  195. $res = self::$wechat_pay->order->unify($config);
  196. \Yii::error($res);
  197. } catch (GuzzleException $e) {
  198. return [
  199. 'code' => 1,
  200. 'msg' => $e->getMessage()
  201. ];
  202. } catch (\Exception $e) {
  203. return [
  204. 'code' => 1,
  205. 'msg' => $e->getMessage()
  206. ];
  207. }
  208. if (!$res) {
  209. return [
  210. 'code' => 1,
  211. 'msg' => '支付失败',
  212. ];
  213. }
  214. if (isset($res['err_code']) && $res['err_code'] == 'INVALID_REQUEST') {
  215. return [
  216. 'code' => 1,
  217. 'msg' => '支付失败,' . $res['err_code_des'] ?? ($res['return_msg'] ?? ''),
  218. 'res' => $res,
  219. ];
  220. }
  221. if ($res['return_code'] != 'SUCCESS') {
  222. return [
  223. 'code' => 1,
  224. 'msg' => '支付失败,' . (isset($res['return_msg']) ? $res['return_msg'] : ''),
  225. 'res' => $res,
  226. ];
  227. }
  228. if ($res['result_code'] != 'SUCCESS') {
  229. if ($res['err_code'] == 'INVALID_REQUEST') { //商户订单号重复
  230. $order->order_no = OrderNo::getOrderNo($type);
  231. $order->save();
  232. if ($type != OrderNo::ORDER_UNION) {
  233. if ($is_app) {
  234. return self::orderUnify($order, $type, $goods_names, 0, true, $balance_price);
  235. } else if ($is_h5) {
  236. return self::orderUnify($order, $type, $goods_names, 0, false, $balance_price, true);
  237. } else if ($is_official) {
  238. return self::orderUnify($order, $type, $goods_names, 0, false, $balance_price, false, true);
  239. } else {
  240. return self::orderUnify($order, $type, $goods_names, 0, false, $balance_price);
  241. }
  242. }
  243. if ($is_app) {
  244. return self::orderUnify($order, $type, '', $total_pay_price, true, $balance_price);
  245. } else if ($is_h5) {
  246. return self::orderUnify($order, $type, '', $total_pay_price, false, $balance_price, true);
  247. } else if ($is_official) {
  248. return self::orderUnify($order, $type, '', $total_pay_price, false, $balance_price, false, true);
  249. } else {
  250. return self::orderUnify($order, $type, '', $total_pay_price, false, $balance_price);
  251. }
  252. } else {
  253. return [
  254. 'code' => 1,
  255. 'msg' => '支付失败,' . (isset($res['err_code_des']) ? $res['err_code_des'] : ''),
  256. 'res' => $res,
  257. ];
  258. }
  259. }
  260. if ($is_app) {
  261. self::$wechat_pay->config->app_id = $store_wechat->app_id;
  262. self::$wechat_pay->config->mch_id = $store_wechat->mch_id;
  263. $pay_data = self::$wechat_pay->jssdk->appConfig($res['prepay_id']);
  264. } else {
  265. $pay_data = self::$wechat_pay->jssdk->bridgeConfig($res['prepay_id'], false);
  266. }
  267. $return = [
  268. 'code' => 0,
  269. 'msg' => 'success',
  270. 'res' => $res,
  271. 'data' => $pay_data
  272. ];
  273. if ($type == OrderNo::ORDER_UNION) {
  274. foreach ($order as $value) {
  275. $value->order_union_id = $order_union->id;
  276. $value->save();
  277. }
  278. $return['order_no'] = $config['out_trade_no'];
  279. $return['body'] = $config['body'];
  280. }
  281. if ($is_h5) {
  282. $return['res']['mweb_url'] = $return['res']['mweb_url'] . '&redirect_url=' . \Yii::$app->request->hostInfo . '/h5/#/order/order/order';
  283. }
  284. \Yii::error($return);
  285. return $return;
  286. }
  287. /**
  288. * 退款
  289. * @param Object $order
  290. * @param string $orderRefundNo
  291. * @param string $type
  292. * @param integer $refundFee
  293. * @param $refund_account
  294. * @return array
  295. */
  296. public static function orderRefund($order, $type, $refundFee, $orderRefundNo, $refund_account = null) {
  297. try {
  298. if ($order->order_origin == Order::ORDER_SOURCE_APP) {
  299. parent::init($order->store_id, $order->is_platform, 0, WechatConfig::TYPE_CONFIG_APP);
  300. } else if ($order->order_origin == Order::ORDER_SOURCE_MP) {
  301. parent::init($order->store_id, $order->is_platform, 0, WechatConfig::TYPE_CONFIG_MP);
  302. } else if ($order->order_origin == Order::ORDER_SOURCE_WEB) {
  303. parent::init($order->store_id, $order->is_platform, 0, WechatConfig::TYPE_CONFIG_H5);
  304. } else {
  305. parent::init($order->store_id, $order->is_platform);
  306. }
  307. } catch (\Exception $e) {
  308. return [
  309. 'code' => 1,
  310. 'msg' => $e->getMessage()
  311. ];
  312. }
  313. if ($type != OrderNo::ORDER_UNION) {
  314. if ($order->is_combine_pay == 1 && $order->combine_money > 0) {
  315. $payPrice = round($order->pay_price - $order->combine_money, 2);
  316. } else {
  317. $payPrice = $order->pay_price;
  318. }
  319. } else {
  320. // 联合订单支付的总额
  321. // $payPrice = $order->price;
  322. }
  323. if ($order->is_combine_pay == 1 && $order->combine_money > 0) {
  324. $refundFee = round($refundFee - $order->combine_money, 2);
  325. }
  326. $data = [
  327. 'out_trade_no' => $order->order_no,
  328. 'out_refund_no' => $orderRefundNo,
  329. 'total_fee' => intval(bcmul($payPrice, 100)),
  330. 'refund_fee' => intval(bcmul($refundFee, 100)),
  331. ];
  332. if (isset($order->order_union_id) && $order->order_union_id != 0) {
  333. // 多商户合并订单退款
  334. $orderUnion = OrderUnion::findOne($order->order_union_id);
  335. if (!$orderUnion) {
  336. return [
  337. 'code' => 1,
  338. 'msg' => '订单取消失败,合并支付订单不存在。',
  339. ];
  340. }
  341. $data['out_trade_no'] = $orderUnion->order_no;
  342. $data['total_fee'] = $orderUnion->price * 100;
  343. }
  344. $config = [];
  345. if ($refund_account) {
  346. $config['refund_account'] = $refund_account;
  347. }
  348. try {
  349. $res = self::$wechat_pay->refund->byOutTradeNumber($data['out_trade_no'], $orderRefundNo, $data['total_fee'],
  350. $data['refund_fee'], $config);
  351. } catch (\Exception $e) {
  352. return [
  353. 'code' => 1,
  354. 'msg' => $e->getMessage()
  355. ];
  356. }
  357. if (!$res) {
  358. return [
  359. 'code' => 1,
  360. 'msg' => '订单取消失败,退款失败,服务端配置出错',
  361. ];
  362. }
  363. if ($res['return_code'] != 'SUCCESS') {
  364. return [
  365. 'code' => 1,
  366. 'msg' => '订单取消失败,退款失败,' . $res['return_msg'],
  367. 'res' => $res,
  368. ];
  369. }
  370. if (isset($res['err_code']) && $res['err_code'] == 'NOTENOUGH' && !$refund_account) {
  371. // 交易未结算资金不足,请使用可用余额退款
  372. return [
  373. 'code' => 1,
  374. 'msg' => '订单取消失败,退款失败,' . $res['err_code_des'],
  375. 'res' => $res,
  376. ];
  377. return self::orderRefund($order, $type, $refundFee, $orderRefundNo, 'REFUND_SOURCE_RECHARGE_FUNDS');
  378. }
  379. if ($res['result_code'] != 'SUCCESS') {
  380. $refundQuery = self::$wechat_pay->order->queryByOutTradeNumber($order->order_no);
  381. if ($refundQuery['return_code'] != 'SUCCESS') {
  382. return [
  383. 'code' => 1,
  384. 'msg' => '订单取消失败,退款失败,' . $refundQuery['return_msg'],
  385. 'res' => $refundQuery,
  386. ];
  387. }
  388. if ($refundQuery['result_code'] == 'FAIL') {
  389. return [
  390. 'code' => 1,
  391. 'msg' => '订单取消失败,退款失败,' . $res['err_code_des'],
  392. 'res' => $res,
  393. ];
  394. }
  395. if ($refundQuery['result_code'] != 'SUCCESS') {
  396. return [
  397. 'code' => 1,
  398. 'msg' => '订单取消失败,退款失败,' . $refundQuery['err_code_des'],
  399. 'res' => $refundQuery,
  400. ];
  401. }
  402. if ($refundQuery['refund_status_0'] != 'SUCCESS') {
  403. return [
  404. 'code' => 1,
  405. 'msg' => '订单取消失败,退款失败,' . $refundQuery['err_code_des'],
  406. 'res' => $refundQuery,
  407. ];
  408. }
  409. }
  410. // 联合支付,退余额
  411. if ($order->is_combine_pay == 1 && $order->combine_money > 0) {
  412. AccountLog::saveLog($order->user_id, $order->combine_money, AccountLog::TYPE_BALANCE, AccountLog::LOG_TYPE_INCOME, AccountLog::TYPE_PLATFORM_REFUND_ORDER, $order->id, "商城订单退款,订单号:{$order->order_no}");
  413. }
  414. return [
  415. 'code' => 0,
  416. 'msg' => 'success',
  417. 'data' => true
  418. ];
  419. }
  420. /**
  421. * 转账
  422. * @param string $trade_no 商户订单号
  423. * @param integer $amount
  424. * @param integer $user_name
  425. * @param string $desc
  426. * @param string $openid
  427. * @return array
  428. * @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException
  429. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  430. * @throws \GuzzleHttp\Exception\GuzzleException
  431. */
  432. public static function transfer($trade_no, $amount, $user_name, $openid, $desc = null) {
  433. if (!$trade_no || $amount <= 0 || empty($amount) || empty($openid)) {
  434. return [
  435. 'code' => 1,
  436. 'msg' => '缺少必要参数'
  437. ];
  438. }
  439. try {
  440. parent::init();
  441. } catch (\Exception $e) {
  442. return [
  443. 'code' => 1,
  444. 'msg' => $e->getMessage()
  445. ];
  446. }
  447. $data = [
  448. 'partner_trade_no' => $trade_no,
  449. 'openid' => $openid,
  450. 'check_name' => 'FORCE_CHECK', // 强校验用户姓名 NO_CHECK:不校验用户姓名
  451. 're_user_name' => $user_name,
  452. 'amount' => $amount * 100,
  453. 'desc' => $desc ? $desc : '转账'
  454. ];
  455. $res = self::$wechat_pay->transfer->toBalance($data);
  456. if ($res['return_code'] != 'SUCCESS') {
  457. return [
  458. 'code' => 1,
  459. 'msg' => '转账失败:,' . $res['return_msg'],
  460. 'res' => $res,
  461. ];
  462. }
  463. if ($res['result_code'] != 'SUCCESS') {
  464. return [
  465. 'code' => 1,
  466. 'msg' => '转账失败:' . $res['err_code_des'],
  467. 'res' => $res,
  468. ];
  469. }
  470. return [
  471. 'code' => 0,
  472. 'msg' => 'success',
  473. 'data' => true
  474. ];
  475. }
  476. /**
  477. * 转账
  478. * @param string $trade_no 商户订单号
  479. * @param integer $amount
  480. * @param integer $user_name
  481. * @param string $desc
  482. * @param string $openid
  483. * @return array
  484. * @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException
  485. * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException
  486. * @throws \GuzzleHttp\Exception\GuzzleException
  487. */
  488. public static function saasTransfer($trade_no, $amount, $user_name, $openid, $desc = null) {
  489. if (!$trade_no || $amount <= 0 || empty($amount) || empty($openid)) {
  490. return [
  491. 'code' => 1,
  492. 'msg' => '缺少必要参数'
  493. ];
  494. }
  495. try {
  496. parent::initSaas(WechatProfit::WECHAT_KIND_PAY);
  497. } catch (\Exception $e) {
  498. return [
  499. 'code' => 1,
  500. 'msg' => $e->getMessage()
  501. ];
  502. }
  503. $data = [
  504. 'partner_trade_no' => $trade_no,
  505. 'openid' => $openid,
  506. 'check_name' => 'NO_CHECK', // FORCE_CHECK:强校验用户姓名, NO_CHECK:不校验用户姓名
  507. 're_user_name' => $user_name,
  508. 'amount' => $amount * 100,
  509. 'desc' => $desc ? $desc : '转账'
  510. ];
  511. $res = self::$wechat_pay->transfer->toBalance($data);
  512. if ($res['return_code'] != 'SUCCESS') {
  513. return [
  514. 'code' => 1,
  515. 'msg' => '转账失败:,' . $res['return_msg'],
  516. 'res' => $res,
  517. ];
  518. }
  519. if ($res['result_code'] != 'SUCCESS') {
  520. return [
  521. 'code' => 1,
  522. 'msg' => '转账失败:' . $res['err_code_des'],
  523. 'res' => $res,
  524. ];
  525. }
  526. return [
  527. 'code' => 0,
  528. 'msg' => 'success',
  529. 'data' => true
  530. ];
  531. }
  532. /**
  533. * 查询订单
  534. * @param $trade_no
  535. * @return array
  536. */
  537. public static function find($trade_no) {
  538. if (!$trade_no) {
  539. return [
  540. 'code' => 1,
  541. 'msg' => '缺少必要参数'
  542. ];
  543. }
  544. try {
  545. parent::init();
  546. } catch (\Exception $e) {
  547. return [
  548. 'code' => 1,
  549. 'msg' => $e->getMessage()
  550. ];
  551. }
  552. try {
  553. $res = self::$wechat_pay->order->queryByOutTradeNumber($trade_no);
  554. if ($res['trade_state'] == 'SUCCESS') {
  555. return [
  556. 'code' => 0,
  557. 'msg' => 'success',
  558. 'res' => $res,
  559. ];
  560. }
  561. } catch (Exception $e) {
  562. \Yii::warning(['WECHAT QUERY ORDER <==========>', $trade_no, $e->getMessage()]);
  563. }
  564. return [
  565. 'code' => 1,
  566. 'msg' => 'fail',
  567. 'data' => false
  568. ];
  569. }
  570. // 微信扫码付款
  571. public static function micropay($order, $type, $goods_names = '赤炎鹰洛龙店-超市', $total_pay_price = 0,$balance_price = 0,$auth_code){
  572. // 服务商参数
  573. $wechat_config = Option::getSaasWechat();
  574. // 特约商户参数
  575. $store_wechat = WechatConfig::findOne(['store_id' => $order->store_id, 'type' => 1]);
  576. $data = [
  577. 'appid' => $wechat_config['sp_appid'],
  578. 'mch_id' => $wechat_config['sp_mch_id'],
  579. 'sub_mch_id' => $store_wechat->mch_id,
  580. 'nonce_str' => self::makeNonceStr(12),
  581. 'body' => $goods_names,
  582. 'out_trade_no' => $order->order_no,
  583. //'total_fee' => (int)$order->pay_price * 100,
  584. 'total_fee' => $balance_price > 0 ? round(floatval($total_pay_price - $balance_price) * 100) : $total_pay_price * 100,
  585. 'spbill_create_ip' => $_SERVER['REMOTE_ADDR'],
  586. 'auth_code' => $auth_code,
  587. ];
  588. // 是否走分账
  589. if (is_profit_sharing()) {
  590. $data['profit_sharing'] = 'Y';
  591. }
  592. $data['sign'] = self::makeSign($data,$wechat_config['sp_key']);
  593. // 请求微信的支付
  594. $data = DataTransform::arrayToXml($data);
  595. $result = cloud_post('https://api.mch.weixin.qq.com/pay/micropay',$data);
  596. $result = DataTransform::xmlToArray($result);
  597. return $result;
  598. }
  599. // 查询扫码付款状态
  600. public static function orderQuery($order){
  601. // 服务商参数
  602. $wechat_config = Option::getSaasWechat();
  603. // 特约商户参数
  604. $store_wechat = WechatConfig::findOne(['store_id' => $order->store_id, 'type' => 1]);
  605. $data = [
  606. 'appid' => $wechat_config['sp_appid'],
  607. 'mch_id' => $wechat_config['sp_mch_id'],
  608. 'sub_mch_id' => $store_wechat->mch_id,
  609. 'out_trade_no' => $order->order_no,
  610. 'nonce_str' => self::makeNonceStr(12),
  611. ];
  612. $data['sign'] = self::makeSign($data,$wechat_config['sp_key']);
  613. // 请求微信的支付
  614. $data = DataTransform::arrayToXml($data);
  615. $result = cloud_post('https://api.mch.weixin.qq.com/pay/orderquery',$data);
  616. return DataTransform::xmlToArray($result);
  617. }
  618. /**
  619. * 生成签名
  620. * @param $data
  621. * @return string
  622. * User: hankaige
  623. * DATE TIME: 2022/12/8 13:46
  624. */
  625. public static function makeSign($data,$pay_key){
  626. // 去空
  627. foreach($data as $key=>$value){
  628. if(empty($value)){
  629. unset($data[$key]);
  630. }
  631. }
  632. // 按ASSII排序
  633. ksort($data);
  634. // 转字符串
  635. $str = '';
  636. foreach($data as $m=>$n){
  637. $str .= $m . '=' . $n . '&';
  638. }
  639. $str .= 'key='.$pay_key;
  640. // 进行
  641. return strtoupper(MD5($str));
  642. }
  643. /**
  644. * 获取随机字符串
  645. * @param $len
  646. * @return string
  647. * User: hankaige
  648. * DATE TIME: 2022/12/8 13:46
  649. */
  650. public static function makeNonceStr($len = 6): string
  651. {
  652. $chars = array(
  653. "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k",
  654. "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v",
  655. "w", "x", "y", "z", "A", "B", "C", "D", "E", "F", "G",
  656. "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R",
  657. "S", "T", "U", "V", "W", "X", "Y", "Z", "0", "1", "2",
  658. "3", "4", "5", "6", "7", "8", "9"
  659. );
  660. $charsLen = count($chars) - 1;
  661. shuffle($chars); // 将数组打乱
  662. $output = "";
  663. for ($i = 0; $i < $len; $i++) {
  664. $output .= $chars[mt_rand(0, $charsLen)];
  665. }
  666. return $output;
  667. }
  668. }