《PHP教程:php支付寶手機網頁支付類實例》要點:
本文介紹了PHP教程:php支付寶手機網頁支付類實例,希望對您有用。如果有疑問,可以聯系我們。
PHP教程本文實例講述了php支付寶手機網頁支付類.分享給大家供大家參考.具體分析如下:
PHP教程此處注意:
PHP教程① 該類是用在Yii框架里面的,沒有去掉一些框架的東西.
② 本類不能不做任何修改而使用.
PHP教程1. PHP代碼部分如下:????
代碼如下:
<?php
namespace weixin\components;
use Yii;
/**
?* 支付寶手機網頁支付
?*
?* @example
?*???? 創建支付哀求
?*???? $params = []; //支付寶文檔中所需的全部參數
?*???? $alipay = new Alipay();
?*???? $alipay->key = ''; //交易安全校驗碼
?*???? $this->alipay->alipay_config = $params;
?*???? $alipay->buildRequest();
?*????
?*???? 驗證異步通知
?*???? $this->alipay->key = ''; //交易安全校驗碼
?*???? $this->alipay->alipay_config = $data; //支付寶異步通知參數
?*???? $this->alipay->verifyNotify();
?*
?* @package Alipay
?* @author Dyllen
?* @since Version 0.2
?*/
class Alipay {
??? /**
???? * 交易安全校驗碼
???? *
???? * @access public
???? * @var string
???? */
??? public $key;
????
??? /**
???? * 哀求參數配置,支付寶接口文檔中所需的參數
???? *
???? * @access public
???? * @var array
???? */
??? public $alipay_config=[];
????
??? /**
???? * HTTPS證書,用于cURL
???? * 默認和本類文件同級目錄的cacert.pem文件
???? *
???? * @access public
???? * @var string
???? */
??? public $credential;
????
??? public $notify_data = null;
????
??? /**
???? * 支付寶即時到賬網關地址
???? */
??? const ALIPAY_GATEWAY = 'https://mapi.alipay.com/gateway.do?';
????
??? /**
???? * HTTPS形式消息驗證地址
???? */
??? const HTTPS_VERIFY_URL = 'https://mapi.alipay.com/gateway.do?service=notify_verify&';
????
??? /**
???? * HTTP形式消息驗證地址
???? */
??? const HTTP_VERIFY_URL = 'http://notify.alipay.com/trade/notify_query.do?';
????
??? /**
???? * 移動網頁支付網關
???? * @var string
???? */
??? const ALIPAY_PAGE_GATEWAY = 'http://wappaygw.alipay.com/service/rest.htm?';
????
????
??? /**
???? * 創建支付包即時到賬哀求url
???? *
???? * @access public
???? * @return void
???? */
??? public function buildRequest() {
??????? $this->alipay_config['sign'] = $this->signData();
??????? return self::ALIPAY_GATEWAY . $this->createQueryString('', true);???????
??? }
????
??? /**
???? * 創建支付寶手機網頁支付鏈接
???? * @return string
???? */
??? public function buildPageUrl()
??? {
??????? $this->alipay_config['sign'] = $this->signData();
??????? $url = self::ALIPAY_PAGE_GATEWAY. $this->createQueryString('');
??????? $response = $this->getHttpResponseGET($url);
??????? $res = $this->parseResponse(trim($response));
??????? //重新組合支付哀求參數
??????? $this->alipay_config['service'] = 'alipay.wap.auth.authAndExecute';
??????? $this->alipay_config['req_data'] = '<auth_and_execute_req><request_token>'.$res['request_token'].'</request_token></auth_and_execute_req>';
????????
??????? $this->alipay_config['sign'] = $this->signData();
??????? return self::ALIPAY_PAGE_GATEWAY. $this->createQueryString('', true);
??? }
????
??? /**
???? * 驗證支付寶異步通知參數合法性
???? *
???? * @access public
???? * @return boolean
???? */
??? public function verifyNotify() {
??????? $param_tmp = $this->filter(); //過濾待簽名數據
??????? if(!isset($this->alipay_config['notify_data'])) {
??????????? return false;
??????? }
??????? $this->notify_data = $this->xmlToArray($this->alipay_config['notify_data']);
??????? $this->alipay_config['notify_id'] = $this->notify_data['notify_id'];
??????? $responseTxt = 'true';
??????? if( !empty( $this->alipay_config['notify_id'] ) ) {
??????????? $responseTxt = $this->getResponse();
??????? }
??????? unset($this->alipay_config['notify_id']);
??????? $txt = 'service=';
??????? $txt .= $this->alipay_config['service'];
??????? $txt .= '&v='.$this->alipay_config['v'];
??????? $txt .= '&sec_id='.$this->alipay_config['sec_id'];
??????? $txt .= '¬ify_data='.$this->alipay_config['notify_data'];
??????? $txt .= $this->key;?????
??????? $sign = md5($txt);
?
??????? if ( preg_match("/true$/i",$responseTxt) && ($sign == $this->alipay_config['sign']) ) {
??????????? return true;
??????? } else {
??????????? return false;
??????? }
??? }
????
??? /**
???? * 解析授權接口返回
???? * @param string $content 授權接口返回的文本數據
???? * @throws \Exception
???? * @return array
???? */
??? private function parseResponse($content) {
??????? parse_str($content, $arr);
??????? $data = isset($arr['res_data']) ? $arr['res_data'] : $arr['res_error'];
??????? $res_data = simplexml_load_string($data);
??????? if(strlen($res_data->request_token) == 0 || strlen($res_data->msg) > 0) {
??????????? throw new \Exception('code:'.$res_data->code.','.$res_data->msg);
??????? }
??????? $arr['request_token'] = $res_data->request_token->__toString();
??????? return $arr;
??? }
????
??? /**
???? * simpleXML對象轉成數組
???? * @param string $xml
???? * @return multitype:NULL
???? */
??? private function xmlToArray($xml)
??? {
??????? $xml_obj = simplexml_load_string($xml, 'SimpleXMLIterator');
??????? $arr = [];
??????? $xml_obj->rewind(); //指針指向第一個元素
??????? while (1) {
??????????? if( ! is_object($xml_obj->current()) )
??????????? {
??????????????? break;
??????????? }
??????????? $arr[$xml_obj->key()] = $xml_obj->current()->__toString();
??????????? $xml_obj->next(); //指向下一個元素
??????? }
??????? return $arr;
??? }
????
??? /**
???? * 簽名數據
???? * 簽名規則:
???? *???? sign和sign_type不參加簽名,需要去掉
???? *???? 對參數數組依據鍵名按照字母順序升序排序
???? *???? 排序完成之后鍵值對用&字符連接,組成URL的查詢字符串形式待簽名字符串,待簽名數據不需用url encoding
???? *???? MD5簽名:私鑰拼接到待簽名字符串的后面,然后用md5對字符串運算,得到32位簽名結果
???? *????
???? * @return string 已簽名數據
???? */
??? private function signData() {
??????? $param_tmp = $this->getSignString(); //待簽名字符串
????????
??????? if( !isset($this->key) ) {
??????????? return FALSE;
??????? }
????????
??????? $sign = '';
????????
??????? //簽名數據
??????? switch ($this->alipay_config['sec_id']) {
??????????? case '001': //rsa
??????????????? $sign = $this->rsaSign($param_tmp);
??????????????? break;
??????????? case 'DES':
??????????????? break;
??????????? default:
??????????????? $sign = $this->md5Sign($param_tmp);
??????? }
????????
??????? return $sign;
??? }
????
??? /**
???? * MD5加密字符串
???? *
???? * @access private
???? * @param string $data 待加密字符串
???? * @return string
???? */
??? private function md5Sign( $data ) {
??????? return md5($data . $this->key);
??? }
????
??? /**
???? * RSA 加密字符串
???? *
???? * @param string $data 待加密字符串
???? * @return string
???? */
??? private function rsaSign( $data ) {
??????? return false;
??? }
????
??? /**
???? * 獲得待簽名數據
???? *
???? * @access private
???? * @return string
???? */
??? private function getSignString() {
??????? $param_tmp = $this->filter(); //過濾待簽名數據
????????
??????? //排序
??????? ksort($param_tmp);
??????? reset($param_tmp);
????????
??????? //創建查詢字符串形式的待簽名數據
??????? return $this->createQueryString($param_tmp);
??? }
????
??? /**
???? * 過濾待簽名數據,去掉sing、sing_type及空值
???? *
???? * @access private
???? * @return array
???? */
??? private function filter() {
??????? $para_filter = array();
??????? foreach($this->alipay_config as $key => $value){
??????????? if($key == "sign" || $key == "sign_type" || empty($value)) continue;
??????????? else $para_filter[$key] = $value;
??????? }
??????? return $para_filter;
??? }
????
??? /**
???? * 用&拼接字符串,形成URL查詢字符串
???? *
???? * @access private
???? * @param array $data
???? * @param boolean $is_encode 是否對值做urlencode
???? * @return string
???? */
??? private function createQueryString($data=NULL, $is_encode=false ) {
??????? $arr = empty($data) ? $this->alipay_config : $data;
??????? $arg = '';
??????? foreach( $arr as $key => $value ) {
??????????? if($is_encode) {
??????????????? $key = urlencode($key);
??????????????? $value = urlencode($value);
??????????? }
??????????? $arg .= $key . '=' . $value . '&';
??????? }
??????? $arg = substr($arg, 0, strlen($arg)-1); //去掉最后一個&
??????? //如果存在轉義字符,那么去掉轉義
??????? if(get_magic_quotes_gpc()) {$arg = stripslashes($arg);}
????????
??????? return $arg;
??? }
????
??? /**
???? * 獲取遠程服務器ATN結果,驗證返回URL
???? *
???? * 驗證結果集:
???? * invalid命令參數不對 出現這個錯誤,請檢測返回處理中partner和key是否為空
???? * true 返回正確信息
???? * false 請檢查防火墻或者是服務器阻止端口問題以及驗證時間是否超過一分鐘
???? *
???? * @access private
???? * @return 服務器ATN結果
???? */
??? private function getResponse() {
??????? //載入支付配置
??????? $config = Yii::$app->params['alipay'];
????????
??????? $transport = strtolower(trim($config['transport']));
??????? $partner = trim($config['partner']);
??????? $veryfy_url = '';
??????? if($transport == 'https') {
??????????? $veryfy_url = self::HTTPS_VERIFY_URL;
??????? }
??????? else {
??????????? $veryfy_url = self::HTTP_VERIFY_URL;
??????? }
??????? $veryfy_url = $veryfy_url."partner=" . $partner . "¬ify_id=" . $this->alipay_config['notify_id'];
??????? $responseTxt = $this->getHttpResponseGET($veryfy_url);
????
??????? return $responseTxt;
??? }
????
??? /**
???? * 取證書,用于cURL的哀求
???? *
???? * @access private
???? * @return string 證書路徑
???? */
??? private function getCr() {
??????? if( ! empty($this->credential) ) {
??????????? return $this->credential;
??????? }
??????? return __DIR__ . DIRECTORY_SEPARATOR .'cacert.pem';
??? }
????
??? /**
???? * 遠程獲取數據,POST模式
???? * 注意:
???? * 1.使用Crul需要修改服務器中php.ini文件的設置,找到php_curl.dll去掉前面的";"就行了
???? * 2.文件夾中cacert.pem是SSL證書請保證其路徑有效,目前默認路徑是:getcwd().'\\cacert.pem'
???? *
???? * @param $url 指定URL完整路徑地址
???? * @param $cacert_url 指定當前工作目錄絕對路徑
???? * @param $para 哀求的數據
???? * @param $input_charset 編碼格式.默認值:空值
???? * return 遠程輸出的數據
???? */
??? private function getHttpResponsePOST($url, $para, $input_charset = '') {
????
??????? if (trim($input_charset) != '') {
??????????? $url = $url."_input_charset=".$input_charset;
??????? }
??????? $curl = curl_init($url);
??????? curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true);//SSL證書認證
??????? curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2);//嚴格認證
??????? curl_setopt($curl, CURLOPT_CAINFO,$this->getCr());//證書地址
??????? curl_setopt($curl, CURLOPT_HEADER, 0 ); // 過濾HTTP頭
??????? curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);// 顯示輸出結果
??????? curl_setopt($curl, CURLOPT_POST,true); // post傳輸數據
??????? curl_setopt($curl, CURLOPT_POSTFIELDS,$para);// post傳輸數據
??????? $responseText = curl_exec($curl);
??????? //var_dump( curl_error($curl) );//如果執行curl過程中出現異常,可打開此開關,以便查看異常內容
??????? curl_close($curl);
????
??????? return $responseText;
??? }
????
??? /**
???? * 遠程獲取數據,GET模式
???? * 注意:
???? * 1.使用Crul需要修改服務器中php.ini文件的設置,找到php_curl.dll去掉前面的";"就行了
???? * 2.文件夾中cacert.pem是SSL證書請保證其路徑有效,目前默認路徑是:getcwd().'\\cacert.pem'
???? *
???? * @param $url 指定URL完整路徑地址
???? * @param $cacert_url 指定當前工作目錄絕對路徑
???? * return 遠程輸出的數據
???? */
??? private function getHttpResponseGET($url) {
??????? $curl = curl_init($url);
??????? curl_setopt($curl, CURLOPT_HEADER, 0 ); // 過濾HTTP頭
??????? curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);// 顯示輸出結果
??????? curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true);//SSL證書認證
??????? curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2);//嚴格認證
??????? curl_setopt($curl, CURLOPT_CAINFO,$this->getCr());//證書地址
??????? $responseText = curl_exec($curl);
??????? //var_dump( curl_error($curl) );exit;//如果執行curl過程中出現異常,可打開此開關,以便查看異常內容
??????? curl_close($curl);
????
??????? return $responseText;
??? }
}
PHP教程2. 使用方法:
代碼如下:
//授權接口哀求參數
$sum = 0.01; //測試用金額
$req_data = '<direct_trade_create_req><subject>充值</subject>';
$req_data .= '<out_trade_no>'.$orderNo.'</out_trade_no>';
$req_data .= '<total_fee>'.$sum.'</total_fee>';
$req_data .= '<call_back_url>'.Url::toRoute(['payment/return'], true).'</call_back_url>';
$req_data .= '<notify_url>'.Url::toRoute(['payment/notify'], true).'</notify_url>';
$req_data .= '<seller_account_name>'.Yii::$app->params['alipay']['seller_email'].'</seller_account_name>';
$req_data .= '</direct_trade_create_req>';
$params = [
??? 'service' => 'alipay.wap.trade.create.direct',
??? 'format' => 'xml',
??? 'v' => '2.0',
??? 'partner' => Yii::$app->params['alipay']['partner'], //合作者省份ID
??? 'req_id' => date('Ymdhis'),
??? 'sec_id' => Yii::$app->params['alipay']['sign_type'],
??? 'req_data' => $req_data,
];
?
$alipay = new Alipay();
$alipay->key = Yii::$app->params['alipay']['key'];
$alipay->alipay_config = $params;
$url = $alipay->buildPageUrl();
$this->redirect($url);
3. 配置示例:
代碼如下:
//支付寶相關配置
'alipay' => [
??????? 'key' => 'XXXXX',? //交易安全校驗碼,用于簽名的32位密鑰
??????? 'transport' => 'https',???????? //消息驗證地址使用訪問方式
??????? 'seller_email' => 'XXXX', //賣家支付寶賬號,即收款賬戶
??????? 'service' => 'create_direct_pay_by_user', //接口名稱
??????? 'partner' => 'XXXX', //合作者省份ID
??????? '_input_charset' => 'utf-8', //參數編碼字符集
??????? 'sign_type' => 'MD5', //簽名方式,不參加簽名,目前只能是MD5
??????? //以下兩個參數沒用
??????? 'notify_url' => '', //服務器異步通知頁面路徑
??????? 'return_url' => '', //頁面跳轉通知頁面路徑
],
PHP教程希望本文所述對大家的php程序設計有所幫助.
《PHP教程:php支付寶手機網頁支付類實例》是否對您有啟發,歡迎查看更多與《PHP教程:php支付寶手機網頁支付類實例》相關教程,學精學透。維易PHP學院為您提供精彩教程。
轉載請注明本頁網址:
http://www.snjht.com/jiaocheng/11887.html