thinkphp微信支付和微信退款

以下代码修改完自己的
appid
商户号
商户密钥
即可进行运行测试

微信支付

	//微信支付
	public function index(){
     
		//接收用户下单信息
		$data = [];
		$data['sorts'] = input('sorts');//套餐分类
		$data['sciencename']=input('sciencename');//景区名称
		$data['price']=input('price');//订单价格
		$data['create_time']=time();//下单时间
		$data['phone']=input('phone');//用户手机号
		$data['userid']=input('userid');//用户id
		$data['menuid']=input('menuid');//套餐id
		$data['scienceid'] = input('scienceid');//景区ID
		$data['num'] = input('num');//景区ID
		
		if(input('type')){
     
			$data['type']=input('type');//1为音频套餐
		}
		
		$user = Db::name('user_info')->where(array('id' => $data['userid']))->field('openid')->find();//获取当前用户openID
       
		
		//发起微信支付,调用统一下单支付接口
		$fee = $data['price'];//支付金额
		$appid =        '用到的appid';//appid.如果是公众号 就是公众号的appid
		$body =         $data['sorts'];
		$mch_id =       '用到的商户号';  //商户号
		$nonce_str =    $this->nonce_str();//随机字符串
		$notify_url =   'https://ht.hongtuzhijian.top/api/order/notifyurl'; //回调的url
		$openid =       $user['openid'];
		$time = time();
		$year = date('Y',$time);
		$rand = rand(000000000,999999999);
		$out_trade_no = $year.$rand;//订单号
		
		//$out_trade_no = $this->order_number($openid);//商户订单号
		$spbill_create_ip = $_SERVER["REMOTE_ADDR"];//服务器的ip;
		$total_fee =    $fee*100;// 微信支付单位是分,所以这里需要*100
		$trade_type = 'JSAPI';//交易类型 默认
		
		$data['order_no']=$out_trade_no;
		//echo json_encode($data);exit;
		$order = Db::name('user_order')->insert($data);
		
	 
		//这里是按照顺序的 因为下面的签名是按照顺序 排序错误 肯定出错
		$post['appid'] = $appid;
		$post['body'] = $body;
		$post['mch_id'] = $mch_id;
		$post['nonce_str'] = $nonce_str;//随机字符串
		$post['notify_url'] = $notify_url;
		$post['openid'] = $openid;
		$post['out_trade_no'] = $out_trade_no;
		$post['spbill_create_ip'] = $spbill_create_ip;//用户终端的ip
		$post['total_fee'] = $total_fee;//总金额 
		$post['trade_type'] = $trade_type;
		
		$sign = $this->sign($post);//签名
		$post_xml = '
			   '.$appid.'
			   '.$body.'
'.$mch_id.'
'.$nonce_str.'
'.$notify_url.'
'.$openid.'
'.$out_trade_no.'
'.$spbill_create_ip.'
'.$total_fee.'
'.$trade_type.'
'.$sign.'
 ';
//print_r($post_xml);die;
//统一接口prepay_id
$url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';//调用统一下单
$xml = $this->http_request($url,$post_xml);
$array = $this->xml($xml);//全要大写
//echo json_encode($array);exit;
//print_r($array);
unset($data);
//统一下单请求成功回调前端支付参数
if($array['RETURN_CODE'] == 'SUCCESS' && $array['RESULT_CODE'] == 'SUCCESS'){

$time = time();
$tmp=[];//临时数组用于签名
$tmp['appId'] = $appid;
$tmp['nonceStr'] = $nonce_str;
$tmp['package'] = 'prepay_id='.$array['PREPAY_ID'];
$tmp['signType'] = 'MD5';
$tmp['timeStamp'] = $time;
$data['state'] = 200;
$data['timeStamp'] = $time;//时间戳
$data['nonceStr'] = $nonce_str;//随机字符串
$data['signType'] = 'MD5';//签名算法,暂支持 MD5
$data['package'] = 'prepay_id='.$array['PREPAY_ID'];//统一下单接口返回的 prepay_id 参数值
$data['paySign'] = $this->sign($tmp);//签名
$data['out_trade_no'] = $out_trade_no;
}else{

$data['state'] = 0;
$data['text'] = "错误";
$data['RETURN_CODE'] = $array['RETURN_CODE'];
$data['RETURN_MSG'] = $array['RETURN_MSG'];
}
echo json_encode($data);
}
//随机32位字符串
private function nonce_str(){

$result = '';
$str = 'QWERTYUIOPASDFGHJKLZXVBNMqwertyuioplkjhgfdsamnbvcxz';
for ($i=0;$i<32;$i++){

$result .= $str[rand(0,48)];
}
return $result;
}
//生成订单号
private function order_number($openid){

//date('Ymd',time()).time().rand(10,99);//18位
return md5($openid.time().rand(10,99));//32位
}
//签名 $data要先排好顺序
private function sign($data){

$stringA = '';
foreach ($data as $key=>$value){

if(!$value) continue;
if($stringA) $stringA .= '&'.$key."=".$value;
else $stringA = $key."=".$value;
}
$wx_key = '用到的key';//申请支付后有给予一个商户账号和密码,登陆后自己设置的key
$stringSignTemp = $stringA.'&key='.$wx_key;
return strtoupper(md5($stringSignTemp));
}
//curl请求
public function http_request($url,$data = null,$headers=array())
{

$curl = curl_init();
if( count($headers) >= 1 ){

curl_setopt($curl, CURLOPT_HTTPHEADER, $headers);
}
curl_setopt($curl, CURLOPT_URL, $url);
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, FALSE);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, FALSE);
if (!empty($data)){

curl_setopt($curl, CURLOPT_POST, 1);
curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
}
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
$output = curl_exec($curl);
curl_close($curl);
return $output;
}
//获取xml
public function xml($xml){

$p = xml_parser_create();
xml_parse_into_struct($p, $xml, $vals, $index);
xml_parser_free($p);
$data = [];
foreach ($index as $key=>$value) {

if($key == 'xml' || $key == 'XML') continue;
$tag = $vals[$value[0]]['tag'];
$value = $vals[$value[0]]['value'];
$data[$tag] = $value;
}
return $data;
}
//支付回调接口
public function notifyurl(){

$res_xml = file_get_contents("php://input");
libxml_disable_entity_loader(true);
$ret = json_decode(json_encode(simplexml_load_string($res_xml,'simpleXMLElement',LIBXML_NOCDATA)),true);
$data = array();
$data['order_sn'] = $ret['out_trade_no'];
$data['trade_no'] = $ret['transaction_id'];
$data['total_fee'] = $ret['total_fee'];
$check_info = DB::name('user_order')->where(array('order_no'=>$data['order_sn']))->find();
if (!$check_info) {

echo json_encode(array('state'=>1,'msg'=>'订单信息错误'));
}
if($ret['return_code'] == 'SUCCESS'){

$up = array();
$up['status'] = 2;
$up['update_time'] = time();
$res = DB::name('user_order')->where(array('order_no'=>$data['order_sn']))->update($up);//更新订单状态
}
// 更新成功 付款成功
if ($res) {

$xml = "";
$xml.="";
echo $xml;
}
}

微信退款

退款需要传订单号和订单金额两个必要参数

  	 /**
     * 默认支付参数配置,可以在这里配置,也可以在初始化的时候,统一传入参数
     * @var array
     */
	private $config = array(
        	'appid'			=> '用到的小程序id', //小程序id
			'mch_id'	 	=> '用到的商户号',//商户号
			'pay_apikey' 	=> '用到的商户key',//商户key
    );
	/**
     * 使用 $this->name=$value 	配置参数
     * @param  string $name 	配置名称
     * @param  string $value    配置值
     */
	public function __set($name,$value){
     
        if(isset($this->config[$name])) {
     
            $this->config[$name] = $value;
        }
    }
	
	/**
     * 使用 $this->name 获取配置
     * @param  string $name 配置名称
     * @return multitype    配置值
     */
    public function __get($name) {
     
        return $this->config[$name];
    }
	
	public function __isset($name){
     
        return isset($this->config[$name]);
    }
	
	//----------------------------------------------------------退款---------------------------------------------------------
	
	/**
     * 微信退款(POST)
     * @param string(28) $out_trade_no      在微信支付的时候,微信服务器生成的订单流水号,在支付通知中有返回
     * @param string $out_refund_no 		同订单号
     * @param string $total_fee 			微信支付的时候支付的总金额(单位:分)
     * @param string $refund_fee 			此次要退款金额(单位:分)
     * @return string						xml格式的数据
     */
	public function refund($row){
     
		$config = $this->config;
		//退款参数
		$refundorder = array(
			'appid'			=> $config['appid'],
			'mch_id'		=> $config['mch_id'],
			'nonce_str'		=> $this->getNonceStr(),
			'out_trade_no'	=> $row['order_no'],
			'out_refund_no' => $row['order_no'],
			'total_fee'		=> $row['price']*100,
			'refund_fee'	=> $row['price']*100
		);
		$refundorder['sign'] = self::makeSign($refundorder);
		//请求数据,进行退款
		$xmldata = self::array2xml($refundorder);
		$url = 'https://api.mch.weixin.qq.com/secapi/pay/refund';
        $res = self::curl_post_ssl($url, $xmldata);
        if(!$res){
     
			return array('status'=>0, 'msg'=>"Can't connect the server" );
        }
		// 这句file_put_contents是用来查看服务器返回的结果 测试完可以删除了
		// file_put_contents('./log3.txt',$res,FILE_APPEND);
		$content = self::xml2array($res);
		// if(strval($content['result_code']) == 'FAIL'){
     
    	//return array('status'=>0, 'msg'=>strval($content['err_code']).':'.strval($content['err_code_des']));
		//     }
		if(strval($content['return_code']) == 'SUCCESS'){
     
			//退款成功
			//自己写退款成功之后要运行的代码
	    }
		// return $content;
	}
	

	
//---------------------------------------------------------------用到的函数------------------------------------------------------
	/**
     * 将一个数组转换为 XML 结构的字符串
     * @param array $arr 要转换的数组
     * @param int $level 节点层级, 1 为 Root.
     * @return string XML 结构的字符串
     */
    protected function array2xml($arr, $level = 1) {
     
        $s = $level == 1 ? "" : '';
        foreach($arr as $tagname => $value) {
     
            if (is_numeric($tagname)) {
     
                $tagname = $value['TagName'];
                unset($value['TagName']);
            }
            if(!is_array($value)) {
     
                $s .= "<{
     $tagname}>".(!is_numeric($value) ? ' : '').$value.(!is_numeric($value) ? ']]>' : '')."$tagname}>";
            } else {
     
                $s .= "<{
     $tagname}>" . $this->array2xml($value, $level + 1)."$tagname}>";
            }
        }
        $s = preg_replace("/([\x01-\x08\x0b-\x0c\x0e-\x1f])+/", ' ', $s);
        return $level == 1 ? $s."" : $s;
    }
	
	/**
	 * 将xml转为array
	 * @param  string 	$xml xml字符串
	 * @return array    转换得到的数组
	 */
	protected function xml2array($xml){
        
		//禁止引用外部xml实体
		libxml_disable_entity_loader(true);
		$result= json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);        
		return $result;
	}
	
	/**
	 * 
	 * 产生随机字符串,不长于32位
	 * @param int $length
	 * @return 产生的随机字符串
	 */
	protected function getNonceStr($length = 32) {
     
		$chars = "abcdefghijklmnopqrstuvwxyz0123456789";  
		$str ="";
		for ( $i = 0; $i < $length; $i++ )  {
       
			$str .= substr($chars, mt_rand(0, strlen($chars)-1), 1);  
		} 
		return $str;
	}
	
	/**
	* 生成签名
	* @return 签名
	*/
	protected function makeSign($data){
     
		//获取微信支付秘钥
		$key = $this->config['pay_apikey'];
		// 去空
		$data=array_filter($data);
		//签名步骤一:按字典序排序参数
		ksort($data);
		$string_a=http_build_query($data);
		$string_a=urldecode($string_a);
		//签名步骤二:在string后加入KEY
		$string_sign_temp=$string_a."&key=".$key;
		//签名步骤三:MD5加密
		$sign = md5($string_sign_temp);
		// 签名步骤四:所有字符转为大写
		$result=strtoupper($sign);
		return $result;
	}
	
	/**
	 * 获取IP地址
	 * @return [String] [ip地址]
	 */
	protected function getip() {
     
        static $ip = '';
        $ip = $_SERVER['REMOTE_ADDR'];
        if(isset($_SERVER['HTTP_CDN_SRC_IP'])) {
     
            $ip = $_SERVER['HTTP_CDN_SRC_IP'];
        } elseif (isset($_SERVER['HTTP_CLIENT_IP']) && preg_match('/^([0-9]{1,3}\.){3}[0-9]{1,3}$/', $_SERVER['HTTP_CLIENT_IP'])) {
     
            $ip = $_SERVER['HTTP_CLIENT_IP'];
        } elseif(isset($_SERVER['HTTP_X_FORWARDED_FOR']) AND preg_match_all('#\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}#s', $_SERVER['HTTP_X_FORWARDED_FOR'], $matches)) {
     
            foreach ($matches[0] AS $xip) {
     
                if (!preg_match('#^(10|172\.16|192\.168)\.#', $xip)) {
     
                    $ip = $xip;
                    break;
                }
            }
        }
        return $ip;
    }
	
	/**
	 * 微信支付退款发起请求
	 */
	protected function curl_post_ssl($url, $xmldata, $second=30,$aHeader=array()){
     
		$config = $this->config;
		
		$ch = curl_init();
		//超时时间
		curl_setopt($ch,CURLOPT_TIMEOUT,$second);
		curl_setopt($ch,CURLOPT_RETURNTRANSFER, 1);
		//这里设置代理,如果有的话
		//curl_setopt($ch,CURLOPT_PROXY, '10.206.30.98');
		//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,0);
		curl_setopt($ch, CURLOPT_SSLVERSION, 1);
		
		$cert = __DIR__.'/cert/apiclient_cert.pem';
		$key = __DIR__.'/cert/apiclient_key.pem';
		//默认格式为PEM,可以注释
		curl_setopt($ch,CURLOPT_SSLCERT, $cert); //这个是证书的位置绝对路径
		curl_setopt($ch,CURLOPT_SSLKEY, $key); //这个也是证书的位置绝对路径
		//curl_setopt($ch,CURLOPT_CAINFO,$config['rootca']);
	 
		if( count($aHeader) >= 1 ){
     
			curl_setopt($ch, CURLOPT_HTTPHEADER, $aHeader);
		}
	 
		curl_setopt($ch,CURLOPT_POST, true);
		curl_setopt($ch,CURLOPT_POSTFIELDS,$xmldata);
		$data = curl_exec($ch);
		if($data){
     
			curl_close($ch);
			return $data;
		}else {
      
			$error = curl_errno($ch);
			echo "call faild, errorCode:$error\n"; 
			curl_close($ch);
			return false;
		}
	}

你可能感兴趣的:(php,小程序,php)