WechatTransferSvc.php 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. <?php
  2. namespace app\service;
  3. use GuzzleHttp\Client;
  4. use GuzzleHttp\HandlerStack;
  5. use WechatPay\GuzzleMiddleware\WechatPayMiddleware;
  6. use WeChatPay\Util\PemUtil;
  7. class WechatTransferSvc
  8. {
  9. public $mchid = '你的商户id';
  10. public $appid = '小程序appid';
  11. public $serial_no = '6B66AFDAF3FC7D4AFDA8A51F61194B0DC8432BE1'; // 证书序列号,不知道怎么获取的看上面的图
  12. private static function filePath($path){
  13. return RUNTIME_PATH.$path;
  14. }
  15. public function __construct()
  16. {
  17. $this->mchid=config('site.mch_id');
  18. $this->appid=config('site.sender_appid');
  19. $this->serial_no=file_get_contents(self::filePath('serial_no'));
  20. }
  21. public static function client(){
  22. static $client;
  23. if(!$client) {
  24. $merchantId = config('pay.wechat.mch_id'); // 商户号
  25. $merchantSerialNumber = config('pay.wechat.serial_no'); // 商户API证书序列号
  26. $merchantPrivateKey = PemUtil::loadPrivateKey(self::filePath('key')); // 商户私钥文件路径
  27. $wechatpayCertificate = PemUtil::loadCertificate(self::filePath('platform.pem')); // 微信支付平台证书文件路径
  28. $wechatpayMiddleware = WechatPayMiddleware::builder()
  29. ->withMerchant($merchantId, $merchantSerialNumber, $merchantPrivateKey) // 传入商户相关配置
  30. ->withWechatPay([$wechatpayCertificate]) // 可传入多个微信支付平台证书,参数类型为array
  31. ->build();
  32. $stack = HandlerStack::create();
  33. $stack->push($wechatpayMiddleware, 'wechatpay');
  34. $client = new Client(['handler' => $stack]);
  35. }
  36. return $client;
  37. }
  38. public function transfer($openid, $trade_no,$detail_no, $money, $desc='提现')
  39. {
  40. $money=$money*100;
  41. $res=self::client()->post($url='https://api.mch.weixin.qq.com/v3/transfer/batches',[
  42. 'json'=>$body=[
  43. "appid" => $this->appid,//appid
  44. "out_batch_no" => $trade_no,//商家批次单号
  45. "batch_name" => $desc,//批次名称
  46. "batch_remark" => $desc,//批次备注
  47. "total_amount" => $money,// 转账金额单位为“分”
  48. "total_num" => 1, // 转账总笔数
  49. "transfer_detail_list" => [
  50. [
  51. 'out_detail_no' => $detail_no,
  52. 'transfer_amount' => $money,
  53. 'transfer_remark' => $desc,
  54. 'openid' => $openid,
  55. ]
  56. ]
  57. ],
  58. 'headers'=>[
  59. 'Accept'=>'application/json',
  60. 'Authorization'=>$this->token($url,$body),
  61. 'Wechatpay-Serial'=>config('pay.wechat.serial_no')
  62. ]
  63. ]);
  64. if($res->getStatusCode()!==200){
  65. throw new \Exception($res->getBody()->getContents());
  66. }
  67. return [
  68. json_decode($res->getBody()->getContents(),true),
  69. [$trade_no,$detail_no],
  70. $money
  71. ];
  72. }
  73. public function checkDetail($trade_no,$detail_no){
  74. $response=self::client()
  75. ->get($url="https://api.mch.weixin.qq.com/v3/transfer/batches/out-batch-no/$trade_no/details/out-detail-no/$detail_no",[
  76. 'headers'=>[
  77. 'Accept'=>'application/json',
  78. 'Authorization'=>$this->token($url,''),
  79. 'Wechatpay-Serial'=>config('pay.wechat.serial_no')
  80. ]
  81. ]);
  82. if($response->getStatusCode()!==200){
  83. throw new \Exception($response->getBody()->getContents());
  84. }
  85. $res=json_decode($response->getBody()->getContents(),true);
  86. return [
  87. $res['detail_status'],
  88. $res
  89. ];
  90. }
  91. public function getCert(){
  92. $client=(new Client)
  93. ->get($url='https://api.mch.weixin.qq.com/v3/certificates',[
  94. 'headers'=>[
  95. 'Authorization'=>$this->token($url,'','GET'),
  96. 'Accept'=>'application/json',
  97. ]
  98. ]);
  99. $arr=json_decode($client->getBody()->getContents(),true);
  100. $cert=$arr['data'][0]??null;
  101. dd($cert);
  102. }
  103. public function token($url,$body,$http_method="POST"){
  104. if(is_array($body)){
  105. $body=json_encode($body,JSON_UNESCAPED_UNICODE);
  106. }
  107. $url_parts = parse_url($url);
  108. $timestamp=time();
  109. $nonce = strtoupper($this->createNonceStr(32));
  110. $canonical_url = $url_parts['path'];
  111. $message = $http_method."\n".
  112. $canonical_url."\n".
  113. $timestamp."\n".
  114. $nonce."\n".
  115. $body."\n";
  116. openssl_sign($message, $raw_sign, $this->getPrivateKey(), 'sha256WithRSAEncryption');
  117. $sign = base64_encode($raw_sign);
  118. $schema = 'WECHATPAY2-SHA256-RSA2048';
  119. $token = sprintf('mchid="%s",nonce_str="%s",timestamp="%d",serial_no="%s",signature="%s"',
  120. $this->mchid, $nonce, $timestamp, $this->serial_no, $sign);
  121. return "$schema $token";
  122. }
  123. /**
  124. * 生成随机32位字符串
  125. * @param $length
  126. * @return string
  127. */
  128. public function createNonceStr($length = 16) { //生成随机16个字符的字符串
  129. $chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
  130. $str = "";
  131. for ($i = 0; $i < $length; $i++) {
  132. $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
  133. }
  134. return $str;
  135. }
  136. /**
  137. * 获取私钥
  138. * @param $filepath
  139. * @return false|resource
  140. */
  141. private function getPrivateKey($filepath = '')
  142. {
  143. if (empty($filepath)) {
  144. //私钥位置
  145. $filepath = self::filePath('key');
  146. }
  147. return openssl_get_privatekey(file_get_contents($filepath));
  148. }
  149. }