public function pay(){
//獲取openid
if($_GET['code']){ //用code獲取openid
$code=$_GET['code'];
$WX_APPID =C('WX_APPID');//appid
$WX_SECRET = C('WX_SECRET');//AppSecret
$url = "https://api.weixin.qq.com/sns/jscode2session?appid=" .$WX_APPID. "&secret=" .$WX_SECRET. "&js_code=" .$code. "&grant_type=authorization_code";
$infos = json_decode(file_get_contents($url));
$openid = $infos->openid;
}
if($_GET['uid']){
$where['id']=$_GET['uid'];
}
$oid=$_GET['oid'];
//$fee = $_GET['price'];
$fee = 0.01;//舉例支付0.01
$time = time();
$out_trade_no=$this->order_number($openid);
$param = array(
'appid' => '',//小程序id
'mch_id'=>'1247127301',//商戶 idC('WX_ID')
'spbill_create_ip'=>$_SERVER['REMOTE_ADDR'],//終端ip
'notify_url'=>'http://bzl.isoft.mobi/index.php/Home/Pay/notify', //回調通知地址
'nonce_str'=> $this->createNoncestr(),//隨機字符串
'out_trade_no'=>$out_trade_no,//商戶訂單編號
'total_fee'=>$fee*100, //總金額
'openid'=>$openid,//用戶openid
'trade_type'=>'JSAPI',//交易類型
'body' =>'XXX消費', //商品信息
);
$aa['id']=$oid;
$da['out_trade_no']=$out_trade_no;
$order =M('order')->where($aa)->save($da); //給訂單裏生成訂單編號
//經過簽名算法計算得出的簽名值,詳見簽名生成算法
$param['sign'] = $this->getSign($param);
//var_dump($param);die;
//將數組內容轉爲xml格式,向微信發出請求
$xmlData = $this->arrayToXml($param);
$xml_result = $this->postXmlCurl($xmlData,'https://api.mch.weixin.qq.com/pay/unifiedorder',60);
// var_dump($xml_result,$xmlData);die;
$array = $this->xmlToArray($xml_result);
if($array['return_code'] == 'SUCCESS' && $array['result_code'] == 'SUCCESS'){
$time = time();
$key=C('WX_KEY');
$tmp = [];//臨時數組用於簽名
$tmp['appId'] = ' ';
$tmp['nonceStr'] = $array['nonce_str'];
$tmp['package'] = 'prepay_id='.$array['prepay_id'];
$tmp['signType'] = 'MD5';
$tmp['timeStamp'] = (string)$time;
$result['nonceStr'] = $array['nonce_str'];//隨機字符串
$result['signType'] = 'MD5';//簽名算法,暫支持 MD5
$result['package'] = 'prepay_id='.$array['prepay_id'];//統一下單接口返回的 prepay_id 參數值,提交格式如:prepay_id=*
$result['paySign'] = $this->getSign($tmp);
$result['timeStamp'] =(string)$time;
$result['appId'] = $array['appid'];
}else{
$result['return_code'] = $array['return_code'];
$result['return_msg'] = $array['return_msg'];
}
echo json_encode($result,JSON_UNESCAPED_UNICODE);
}php
//生成訂單號
private function order_number($openid){
//date('Ymd',time()).time().rand(10,99);//18位
return md5($openid.time().rand(10,99));//32位
}
/*
* 對要發送到微信統一下單接口的數據進行簽名
*/
protected function getSign($Obj){
foreach ($Obj as $k => $v){
$param[$k] = $v;
}
//簽名步驟一:按字典序排序參數
ksort($param);
$String = $this->formatBizQueryParaMap($param, false);
//簽名步驟二:在string後加入KEY
$wx_key=C('WX_KEY'); //申請支付後有給予一個商戶帳號和密碼,登錄後本身設置的key
$String = $String."&key=".$wx_key;
//簽名步驟三:MD5加密
$String = md5($String);
//簽名步驟四:全部字符轉爲大寫
$result_ = strtoupper($String);
// var_dump($result_);
return $result_;
}
/*
*排序並格式化參數方法,簽名時須要使用
*/
protected function formatBizQueryParaMap($paraMap, $urlencode){
$buff = "";
ksort($paraMap);
foreach ($paraMap as $k => $v){
if($urlencode){
$v = urlencode($v);
}
//$buff .= strtolower($k) . "=" . $v . "&";
$buff .= $k . "=" . $v . "&";
}
$reqPar = "";
if (strlen($buff) > 0){
$reqPar = substr($buff, 0, strlen($buff)-1);
}
return $reqPar;
}
/*
* 生成隨機字符串方法
*/
protected function createNoncestr($length = 32 ){
$chars = "abcdefghijklmnopqrstuvwxyz0123456789";
$str ="";
for ( $i = 0; $i < $length; $i++ ) {
$str.= substr($chars, mt_rand(0, strlen($chars)-1), 1);
}
return $str;
}
//數組轉字符串方法
protected 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字符串轉換爲數組
protected static function xmlToArray($xml){
$array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
return $array_data;
}
//發送xml請求方法
private static function postXmlCurl($xml, $url, $second = 30) {
$ch = curl_init();
//設置超時
curl_setopt($ch, CURLOPT_TIMEOUT, $second);
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_setopt($ch, CURLOPT_CONNECTTIMEOUT, 20);
curl_setopt($ch, CURLOPT_TIMEOUT, 40);
set_time_limit(0);
//運行curl
$data = curl_exec($ch);
//返回結果
if ($data) {
curl_close($ch);
return $data;
}else {
$error = curl_errno($ch);
curl_close($ch);
throw new WxPayException("curl出錯,錯誤碼:$error");
}
}算法