WxPay.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. <?php
  2. namespace app\common\library;
  3. use app\common\model\Config;
  4. use think\Controller;
  5. use think\Facade;
  6. use EasyWeChat\Factory;
  7. class WxPay extends Controller
  8. {
  9. /*
  10. 配置参数
  11. */
  12. public function __construct()
  13. {
  14. // $this->appid = Config::get_values('small_wechat_id');
  15. $this->appid = Config::get_values('wechat_appid');
  16. $this->api_key = Config::get_values('wechat_mch_key');
  17. $this->mch_id = Config::get_values('wechat_mch_id');
  18. }
  19. // private $config = array(
  20. // 'appid' => "wx57f0ba125c0f9351" ,//"wxcf1dded808489e2c", /*微信开放平台上的应用id*/
  21. // 'mch_id' => "1526848461",//"1440493402", /*微信申请成功之后邮件中的商户id*/
  22. // 'api_key' => "quanminchuangquanminchuang123521" /*在微信商户平台上自己设定的api密钥 32位*/
  23. // );
  24. public function retrunconfig(){
  25. $config = [
  26. // 'app_id' => Config::get_values('small_wechat_id'),
  27. 'app_id' => Config::get_values('wechat_appid'),
  28. 'mch_id' =>Config::get_values('wechat_mch_id'),
  29. 'key' =>Config::get_values('wechat_mch_key'),
  30. 'sub_mch_id'=>'1612529165'
  31. ];
  32. return $config;
  33. }
  34. public function retrunconfig2(){
  35. $config = [
  36. 'app_id' => Config::get_values('wechat_appid'),
  37. 'mch_id' =>Config::get_values('wechat_mch_id'),
  38. 'key' =>Config::get_values('wechat_mch_key'),
  39. 'sub_mch_id'=>'1612529165'
  40. ];
  41. return $config;
  42. }
  43. //获取预支付订单
  44. public function getPrePayOrder($body, $out_trade_no, $total_fee, $notify_url,$come,$openid){
  45. $url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
  46. $onoce_str = $this->getRandChar(32);
  47. $data["appid"] = $this->appid;
  48. $data["body"] = $body;
  49. $data["mch_id"] = $this->mch_id;
  50. $data["nonce_str"] = $onoce_str;
  51. $data["notify_url"] = $notify_url;
  52. $data["out_trade_no"] = $out_trade_no;
  53. $data["spbill_create_ip"] = $this->get_client_ip();
  54. $data["total_fee"] = floatval($total_fee);
  55. //$data["trade_type"] = $come=='weixin' ? "JSAPI" : "APP";
  56. $data["trade_type"] = "APP";
  57. $data['sign_type'] = "MD5";
  58. // if ($come=='weixin'){
  59. // $data['openid'] = $openid;
  60. // }
  61. $s = $this->getSign($data);
  62. $data["sign"] = $s;
  63. $xml = $this->arrayToXml($data);
  64. $response = $this->postXmlCurl($xml, $url);
  65. //将微信返回的结果xml转成数组
  66. //return $this->xmlstr_to_array($response);
  67. return $this->xmlToArray($response);
  68. }
  69. public function wxrefund($out_trade_no,$out_refund_no,$total_fee,$refund_fee)
  70. {
  71. $param = array(
  72. 'appid'=> $this->appid,
  73. 'mch_id'=> $this->mch_id,
  74. 'nonce_str'=> $this->getRandChar(32),
  75. 'out_trade_no'=>$out_trade_no,
  76. 'out_refund_no'=>$out_refund_no,//订单号
  77. // 'transaction_id'=> $transactionId,//微信订单号
  78. 'total_fee'=> $total_fee*100,
  79. 'refund_fee'=>$refund_fee*100
  80. );
  81. $param['sign'] = $this->getSign($param);
  82. $xmldata = $this->arrayToXml($param);
  83. $xmlresult = $this->postXmlSSLCurl($xmldata,'https://api.mch.weixin.qq.com/secapi/pay/refund');
  84. $result = $this->xmlToArray($xmlresult);
  85. return $result;
  86. }
  87. //执行第二次签名,才能返回给客户端使用
  88. public function getOrder($prepayId){
  89. $data["appid"] = $this->appid;
  90. $data["noncestr"] = $this->getRandChar(32);
  91. $data["package"] = "Sign=WXPay";
  92. // $data['package'] = "prepay_id=".$prepayId;
  93. $data["partnerid"] = $this->mch_id;
  94. $data["prepayid"] = $prepayId;
  95. $data["signType"] = "MD5";
  96. $data["timestamp"] = '' . time() . '';
  97. $s = $this->getSign($data, false);
  98. $data["sign"] = $s;
  99. return $data;
  100. }
  101. /*
  102. 生成签名
  103. */
  104. function getSign($Obj)
  105. {
  106. foreach ($Obj as $k => $v)
  107. {
  108. $Parameters[strtolower($k)] = $v;
  109. }
  110. //签名步骤一:按字典序排序参数
  111. ksort($Parameters);
  112. $String = $this->formatBizQueryParaMap($Parameters, false);
  113. //echo "【string】 =".$String."</br>";
  114. //签名步骤二:在string后加入KEY
  115. $String = $String."&key=".$this->api_key;
  116. // echo "<textarea style='width: 50%; height: 150px;'>$String</textarea> <br />";
  117. //签名步骤三:MD5加密
  118. $result_ = strtoupper(md5($String));
  119. return $result_;
  120. }
  121. //获取指定长度的随机字符串
  122. function getRandChar($length){
  123. $str = null;
  124. $strPol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";
  125. $max = strlen($strPol)-1;
  126. for($i=0;$i<$length;$i++){
  127. $str.=$strPol[rand(0,$max)];//rand($min,$max)生成介于min和max两个数之间的一个随机整数
  128. }
  129. return $str;
  130. }
  131. //数组转xml
  132. function arrayToXml($arr)
  133. {
  134. $xml = "<xml>";
  135. foreach ($arr as $key=>$val)
  136. {
  137. if (is_numeric($val))
  138. {
  139. $xml.="<".$key.">".$val."</".$key.">";
  140. }
  141. else
  142. $xml.="<".$key."><![CDATA[".$val."]]></".$key.">";
  143. }
  144. $xml.="</xml>";
  145. return $xml;
  146. }
  147. //post https请求,CURLOPT_POSTFIELDS xml格式
  148. function postXmlCurl($xml,$url,$second=30)
  149. {
  150. //初始化curl
  151. $ch = curl_init();
  152. //超时时间
  153. curl_setopt($ch,CURLOPT_TIMEOUT,$second);
  154. //这里设置代理,如果有的话
  155. //curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
  156. //curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
  157. curl_setopt($ch,CURLOPT_URL, $url);
  158. curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
  159. curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);
  160. //设置header
  161. curl_setopt($ch, CURLOPT_HEADER, FALSE);
  162. //要求结果为字符串且输出到屏幕上
  163. curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
  164. //post提交方式
  165. curl_setopt($ch, CURLOPT_POST, TRUE);
  166. curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
  167. //运行curl
  168. $data = curl_exec($ch);
  169. //返回结果
  170. if($data)
  171. {
  172. curl_close($ch);
  173. return $data;
  174. }
  175. else
  176. {
  177. $error = curl_errno($ch);
  178. echo "curl出错,错误码:$error"."<br>";
  179. echo "<a href='http://curl.haxx.se/libcurl/c/libcurl-errors.html'>错误原因查询</a></br>";
  180. curl_close($ch);
  181. return false;
  182. }
  183. }
  184. //需要使用证书的请求
  185. private function postXmlSSLCurl($xml,$url,$second=30)
  186. {
  187. $ch = curl_init();
  188. //超时时间
  189. curl_setopt($ch,CURLOPT_TIMEOUT,$second);
  190. //这里设置代理,如果有的话
  191. //curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
  192. //curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
  193. curl_setopt($ch,CURLOPT_URL, $url);
  194. curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
  195. curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);
  196. //设置header
  197. curl_setopt($ch,CURLOPT_HEADER,FALSE);
  198. //要求结果为字符串且输出到屏幕上
  199. curl_setopt($ch,CURLOPT_RETURNTRANSFER,TRUE);
  200. //设置证书
  201. //使用证书:cert 与 key 分别属于两个.pem文件
  202. //默认格式为PEM,可以注释
  203. curl_setopt($ch,CURLOPT_SSLCERTTYPE,'PEM');
  204. curl_setopt($ch,CURLOPT_SSLCERT, dirname(__DIR__) . '/wxcert/apiclient_cert.pem');
  205. //默认格式为PEM,可以注释
  206. curl_setopt($ch,CURLOPT_SSLKEYTYPE,'PEM');
  207. curl_setopt($ch,CURLOPT_SSLKEY, dirname(__DIR__) . '/wxcert/apiclient_key.pem');
  208. //post提交方式
  209. curl_setopt($ch,CURLOPT_POST, true);
  210. curl_setopt($ch,CURLOPT_POSTFIELDS,$xml);
  211. $data = curl_exec($ch);
  212. //返回结果
  213. if($data){
  214. curl_close($ch);
  215. return $data;
  216. }
  217. else {
  218. $error = curl_errno($ch);
  219. echo "curl出错,错误码:$error"."<br>";
  220. curl_close($ch);
  221. return false;
  222. }
  223. }
  224. /*
  225. 获取当前服务器的IP
  226. */
  227. function get_client_ip()
  228. {
  229. if ($_SERVER['REMOTE_ADDR']) {
  230. $cip = $_SERVER['REMOTE_ADDR'];
  231. } elseif (getenv("REMOTE_ADDR")) {
  232. $cip = getenv("REMOTE_ADDR");
  233. } elseif (getenv("HTTP_CLIENT_IP")) {
  234. $cip = getenv("HTTP_CLIENT_IP");
  235. } else {
  236. $cip = "unknown";
  237. }
  238. return $cip;
  239. }
  240. //将数组转成uri字符串
  241. function formatBizQueryParaMap($paraMap, $urlencode)
  242. {
  243. $buff = "";
  244. ksort($paraMap);
  245. foreach ($paraMap as $k => $v)
  246. {
  247. if($urlencode)
  248. {
  249. $v = urlencode($v);
  250. }
  251. $buff .= strtolower($k) . "=" . $v . "&";
  252. }
  253. $reqPar;
  254. if (strlen($buff) > 0)
  255. {
  256. $reqPar = substr($buff, 0, strlen($buff)-1);
  257. }
  258. return $reqPar;
  259. }
  260. /**
  261. xml转成数组
  262. */
  263. // function xmlstr_to_array($xmlstr) {
  264. // $doc = new DOMDocument();
  265. // $doc->loadXML($xmlstr);
  266. // return $this->domnode_to_array($doc->documentElement);
  267. // }
  268. function domnode_to_array($node) {
  269. $output = array();
  270. switch ($node->nodeType) {
  271. case XML_CDATA_SECTION_NODE:
  272. case XML_TEXT_NODE:
  273. $output = trim($node->textContent);
  274. break;
  275. case XML_ELEMENT_NODE:
  276. for ($i=0, $m=$node->childNodes->length; $i<$m; $i++) {
  277. $child = $node->childNodes->item($i);
  278. $v = $this->domnode_to_array($child);
  279. if(isset($child->tagName)) {
  280. $t = $child->tagName;
  281. if(!isset($output[$t])) {
  282. $output[$t] = array();
  283. }
  284. $output[$t][] = $v;
  285. }
  286. elseif($v) {
  287. $output = (string) $v;
  288. }
  289. }
  290. if(is_array($output)) {
  291. if($node->attributes->length) {
  292. $a = array();
  293. foreach($node->attributes as $attrName => $attrNode) {
  294. $a[$attrName] = (string) $attrNode->value;
  295. }
  296. $output['@attributes'] = $a;
  297. }
  298. foreach ($output as $t => $v) {
  299. if(is_array($v) && count($v)==1 && $t!='@attributes') {
  300. $output[$t] = $v[0];
  301. }
  302. }
  303. }
  304. break;
  305. }
  306. return $output;
  307. }
  308. // public function xmltoarray( $xml )
  309. // {
  310. // $arr = $this->xml_to_array($xml);
  311. // $key = array_keys($arr);
  312. // return $arr[$key[0]];
  313. // }
  314. /**
  315. * 作用:将xml转为array
  316. */
  317. public function xmlToArray($xml)
  318. {
  319. //将XML转为array
  320. $array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
  321. return $array_data;
  322. }
  323. public function xml_to_array( $xml )
  324. {
  325. $reg = "/<(\\w+)[^>]*?>([\\x00-\\xFF]*?)<\\/\\1>/";
  326. if(preg_match_all($reg, $xml, $matches))
  327. {
  328. $count = count($matches[0]);
  329. $arr = array();
  330. for($i = 0; $i < $count; $i++)
  331. {
  332. $key = $matches[1][$i];
  333. $val = $this->xml_to_array( $matches[2][$i] ); // 递归
  334. if(array_key_exists($key, $arr))
  335. {
  336. if(is_array($arr[$key]))
  337. {
  338. if(!array_key_exists(0,$arr[$key]))
  339. {
  340. $arr[$key] = array($arr[$key]);
  341. }
  342. }else{
  343. $arr[$key] = array($arr[$key]);
  344. }
  345. $arr[$key][] = $val;
  346. }else{
  347. $arr[$key] = $val;
  348. }
  349. }
  350. return $arr;
  351. }else{
  352. return $xml;
  353. }
  354. }
  355. }