<?php
class wxPay
{
private $config = array(
'appid' => "", /*微信开放平台上的应用id*/
'mch_id' => "", /*微信申请成功之后邮件中的商户id*/
'api_key' => "", /*在微信商户平台上自己设定的api密钥 32位*/
'notify_url' => '' /*自定义的回调程序地址*/
);
//下单
public function getPrePayOrder($body, $out_trade_no, $total_fee){
$url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
$notify_url = $this->config["notify_url"];
$onoce_str = $this->createNoncestr();
$data["appid"] = $this->config["appid"];
$data["body"] = $body;
$data["mch_id"] = $this->config['mch_id'];
$data["nonce_str"] = $onoce_str;
$data["notify_url"] = $notify_url;
$data["out_trade_no"] = $out_trade_no;
$data["spbill_create_ip"] = $this->get_client_ip();
$data["total_fee"] = $total_fee;
$data["trade_type"] = "APP";
$sign = $this->getSign($data);
//return $sign;
$data["sign"] = $sign;
$xml = $this->arrayToXml($data);
$response = $this->postXmlCurl($xml, $url); //将微信返回的结果xml转成数组
$response = $this->xmlToArray($response);
if($response['result_code']=='SUCCESS' && $response['return_code']=='SUCCESS'){
$response = $this->getOrder($response['prepay_id']);
return $response;
//return ['code'=>1,'msg'=>'调用成功!','data'=>$response];
}else{
return ['code'=>0,'msg'=>$response['err_code_des']]; } //返回数据 return $response;
}
/** * 作用:产生随机字符串,不长于32位 */
public function createNoncestr( $length = 32 ){
$str = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';//62个字符
$str = str_shuffle($str);
$str = substr($str,0,32);
return $str;
}
//数组转xml
public function arrayToXml($arr){
$xml = "<xml>";
foreach ($arr as $key=>$val){
if (is_numeric($val)){
$xml.="<".$key.">".$val."</".$key.">";
}else{
$xml.="<".$key."><![CDATA[".$val."]]></".$key.">";
}
}
$xml.="</xml>";
return $xml;
}
/** * 作用:将xml转为array */
public function xmlToArray($xml){
//将XML转为array
$array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
return $array_data;
}
public function postXmlCurl($xml,$url,$second=30){
//初始化
$ch = curl_init();
//设置超时
curl_setopt($ch, CURLOPT_TIMEOUT, $second);
//这里设置代理,如果有的话
//curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
//curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
curl_setopt($ch,CURLOPT_URL, $url);
curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);
//设置header
curl_setopt($ch, CURLOPT_HEADER, FALSE);
//要求结果为字符串且输出到屏幕上
curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
//post提交方式
curl_setopt($ch, CURLOPT_POST, TRUE);
curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
//运行curl
$data = curl_exec($ch);
//返回结果
if($data){
curl_close($ch);
return $data;
}else{
$error = curl_errno($ch);
echo "curl出错,错误码:$error"."";
curl_close($ch);
return false;
}
}
public function get_client_ip(){
if ($_SERVER['REMOTE_ADDR']) {
$cip = $_SERVER['REMOTE_ADDR'];
} elseif (getenv("REMOTE_ADDR")) {
$cip = getenv("REMOTE_ADDR");
} elseif (getenv("HTTP_CLIENT_IP")) {
$cip = getenv("HTTP_CLIENT_IP");
} else {
$cip = "unknown";
}
return $cip;
}
public function formatBizQueryParaMap($paraMap, $urlencode){
$buff = "";
ksort($paraMap);
foreach ($paraMap as $k => $v){
if($urlencode){
$v = urlencode($v);
}
$buff .= $k . "=" . $v . "&";
}
if (strlen($buff) > 0){
$reqPar = substr($buff, 0, strlen($buff)-1);
}
return $reqPar;
}
//执行第二次签名,才能返回给客户端使用
public function getOrder($prepayId){
$data["appid"] = $this->config["appid"];
$data["noncestr"] = $this->createNoncestr(32);
$data["package"] = "Sign=WXPay";
$data["partnerid"] = $this->config['mch_id'];
$data["prepayid"] = $prepayId;
$data["timestamp"] = time();
$s = $this->getSign($data, false);
$data["sign"] = $s;
return $data;
}
/*生成签名*/
public function getSign($Obj){
foreach ($Obj as $k => $v){
$Parameters[$k] = $v;
}
//签名步骤一:按字典序排序参数
ksort($Parameters);
$String = $this->formatBizQueryParaMap($Parameters, false);
//echo '【string1】'.$String.'';
//签名步骤二:在string后加入KEY
$String = $String."&key=".$this->config['api_key'];
//return $String;
//echo "【string2】".$String."";
//签名步骤三:MD5加密
$String = md5($String);
//echo "【string3】 ".$String."";
//签名步骤四:所有字符转为大写
$result_ = strtoupper($String);
//echo "【result】 ".$result_."";
return $result_;
}
public function wxPayBack(){
$input=file_get_contents("php://input");
$xml=simplexml_load_string($input);
Log::write($input);
$result_code=(string)$xml->result_code;
$out_trade_no=(string)$xml->out_trade_no;
if($result_code=='SUCCESS'){
//订单数据处理
echo "Success";
}else{
echo "Fail";
}
}
}
调用
function wxPay($order){
require_once(APPPATH . "libraries/wxPay.php");
$wx = new wxPay();
$body = '商品支付';
$out_trade_no = $order;
$total_fee = '1';
//$order['order_amount']
$data = $wx->getPrePayOrder($body,$out_trade_no,$total_fee);
if($data){
return $data;
}else{
return false;
}
}
elseif($pay_type == 3){
$result = $this->wxPay($order_info['order_sn']);
$data = array(
'result' => $result,
'order_sn' => $order_info['order_sn']
);
return_json('1', '请求成功', $data)
回调接收回调数据
$xml = $GLOBALS['HTTP_RAW_POST_DATA'];
if ($xml) {
$array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)));
file_put_contents("text.txt", var_export($array_data, true) . "\r\n", FILE_APPEND);
$out_trade_no = $array_data->out_trade_no;
$trade_no = $array_data->transaction_id;