123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448 |
- <?php
- namespace fast\payment;
- use DOMDocument;
- use Exception;
- /**
- * 支付宝
- * @link https://github.com/mytharcher/alipay-php-sdk
- */
- class Alipay
- {
- const SERVICE = 'create_direct_pay_by_user';
- const SERVICE_WAP = 'alipay.wap.trade.create.direct';
- const SERVICE_WAP_AUTH = 'alipay.wap.auth.authAndExecute';
- const SERVICE_APP = 'mobile.securitypay.pay';
- const GATEWAY = 'https://mapi.alipay.com/gateway.do?';
- const GATEWAY_MOBILE = 'http://wappaygw.alipay.com/service/rest.htm?';
- const VERIFY_URL = 'http://notify.alipay.com/trade/notify_query.do?';
- const VERIFY_URL_HTTPS = 'https://mapi.alipay.com/gateway.do?service=notify_verify&';
- // 配置信息在实例化时传入,以下为范例
- private $config = array(
- // 即时到账方式
- 'payment_type' => 1,
- // 传输协议
- 'transport' => 'http',
- // 编码方式
- 'input_charset' => 'utf-8',
- // 签名方法
- 'sign_type' => 'MD5',
- // 证书路径
- 'cacert' => './cacert.pem',
- //验签公钥地址
- 'public_key_path' => './alipay_public_key.pem',
- 'private_key_path' => '',
- // 支付完成异步通知调用地址
- // 'notify_url' => 'http://'.$_SERVER['HTTP_HOST'].'/order/callback_alipay/notify',
- // 支付完成同步返回地址
- // 'return_url' => 'http://'.$_SERVER['HTTP_HOST'].'/order/callback_alipay/return',
- // 支付宝商家 ID
- 'partner' => '2088xxxxxxxx',
- // // 支付宝商家 KEY
- 'key' => 'xxxxxxxxxxxx',
- // // 支付宝商家注册邮箱
- 'seller_email' => 'email@domain.com'
- );
- private $is_mobile = FALSE;
- public $service = self::SERVICE;
- public $gateway = self::GATEWAY;
- /**
- * 配置
- * @param $options array 配置信息
- * @param null $type string 类型 wap app
- */
- public function __construct($options = [], $type = null)
- {
- if ($config = Config::get('payment.alipay'))
- {
- $this->config = array_merge($this->config, $config);
- }
- $this->config = array_merge($this->config, is_array($options) ? $options : []);
- $this->is_mobile = (($type == 'wap' || $type === true) ? true : false);
- if ($this->is_mobile)
- {
- $this->gateway = self::GATEWAY_MOBILE;
- }
- if ($type == 'wap' || $type === true)
- {
- $this->service = self::SERVICE_WAP;
- }
- elseif ($type == 'app')
- {
- $this->service = self::SERVICE_APP;
- }
- }
- /**
- * 生成请求参数的签名
- *
- * @param $params <Array>
- * @return <String>
- *
- */
- function signParameters($params)
- {
- // 支付宝的签名串必须是未经过 urlencode 的字符串
- // 不清楚为何 PHP 5.5 里没有 http_build_str() 方法
- $paramStr = urldecode(http_build_query($params));
- switch (strtoupper(trim($this->config['sign_type'])))
- {
- case "MD5" :
- $result = md5($paramStr . $this->config['key']);
- break;
- case "RSA" :
- case "0001" :
- $priKey = file_get_contents($this->config['private_key_path']);
- $res = openssl_get_privatekey($priKey);
- openssl_sign($paramStr, $sign, $res);
- openssl_free_key($res);
- //base64编码
- $result = base64_encode($sign);
- break;
- default :
- $result = "";
- }
- return $result;
- }
- /**
- * 准备签名参数
- *
- * @param $params <Array>
- * $params['out_trade_no'] 唯一订单编号
- * $params['subject']
- * $params['total_fee']
- * $params['body']
- * $params['show_url']
- * $params['anti_phishing_key']
- * $params['exter_invoke_ip']
- * $params['it_b_pay']
- * $params['_input_charset']
- * @return <Array>
- */
- function prepareParameters($params)
- {
- $default = array(
- 'service' => $this->service,
- 'partner' => $this->config['partner'],
- '_input_charset' => trim(strtolower($this->config['input_charset']))
- );
- if (!$this->is_mobile)
- {
- $default = array_merge($default, array(
- 'payment_type' => $this->config['payment_type'],
- 'seller_id' => $this->config['partner'],
- 'notify_url' => $this->config['notify_url'],
- ));
- if (isset($this->config['return_url']))
- {
- $default['return_url'] = $this->config['return_url'];
- }
- }
- $params = $this->filterSignParameter(array_merge($default, (array) $params));
- ksort($params);
- reset($params);
- return $params;
- }
- /**
- * 生成签名后的请求参数
- *
- */
- function buildSignedParameters($params)
- {
- $params = $this->prepareParameters($params);
- $params['sign'] = $this->signParameters($params);
- if ($params['service'] != self::SERVICE_WAP && $params['service'] != self::SERVICE_WAP_AUTH)
- {
- $params['sign_type'] = strtoupper(trim($this->config['sign_type']));
- }
- return $params;
- }
- /**
- * https://doc.open.alipay.com/doc2/detail.htm?spm=a219a.7629140.0.0.NgdeQA&treeId=59&articleId=103663&docType=1
- * 服务端生成app支付使用的参数以及签名
- * @param $params <Array>
- * @return <Array>
- */
- function buildSignedParametersForApp($params)
- {
- $params = $this->prepareParameters($params);
- $params['sign'] = urlencode($this->signParameters($params));
- $params['sign_type'] = 'RSA';
- $paramStr = [];
- foreach ($params as $k => &$param)
- {
- $param = '"' . $param . '"';
- $paramStr[] = $k . '=' . $param;
- }
- return implode('&', $paramStr);
- }
- /**
- * 生成请求参数的发送表单HTML
- *
- * 其实这个函数没有必要,更应该使用签名后的参数自己组装,只不过有时候方便就从官方 SDK 里留下了。
- *
- * @param $params <Array> 请求参数(未签名的)
- * @param $method <String> 请求方法,默认:post,可选 get
- * @param $target <String> 提交目标,默认:_self
- * @return <String>
- *
- */
- function buildRequestFormHTML($params, $method = 'post', $target = '_self')
- {
- $params = $this->buildSignedParameters($params);
- $html = '<meta charset="' . $this->config['input_charset'] . '" /><form id="alipaysubmit" name="alipaysubmit" action="' . $this->gateway . ' _input_charset="' . trim(strtolower($this->config['input_charset'])) . '" method="' . $method . ' target="$target">';
- foreach ($params as $key => $value)
- {
- $html .= "<input type='hidden' name='$key' value='$value'/>";
- }
- $html .= "</form><script>document.forms['alipaysubmit'].submit();</script>";
- return $html;
- }
- /**
- * 准备移动网页支付的请求参数
- *
- * 移动网页支付接口不同,需要先服务器提交一次请求,拿到返回 token 再返回客户端发起真实支付请求。
- * 该方法只完成第一次服务端请求,生成参数后需要客户端另行处理(可调用`buildRequestFormHTML`生成表单提交)。
- *
- * @param $params <Array>
- * $params['out_trade_no'] 订单唯一编号
- * $params['subject'] 商品标题
- * $params['total_fee'] 支付总费用
- * $params['merchant_url'] 商品链接地址
- * $params['req_id'] 请求唯一 ID
- * $params['it_b_pay'] 超期时间(秒)
- * @return <Array>/<NULL>
- */
- function prepareMobileTradeData($params)
- {
- // 不要用 SimpleXML 来构建 xml 结构,因为有第一行文档申明支付宝验证不通过
- $xml_str = '<direct_trade_create_req>' .
- '<notify_url>' . $this->config['notify_url'] . '</notify_url>' .
- '<call_back_url>' . $this->config['return_url'] . '</call_back_url>' .
- '<seller_account_name>' . $this->config['seller_email'] . '</seller_account_name>' .
- '<out_trade_no>' . $params['out_trade_no'] . '</out_trade_no>' .
- '<subject>' . htmlspecialchars($params['subject'], ENT_XML1, 'UTF-8') . '</subject>' .
- '<total_fee>' . $params['total_fee'] . '</total_fee>' .
- '<merchant_url>' . $params['merchant_url'] . '</merchant_url>' .
- (isset($params['it_b_pay']) ? '<pay_expire>' . $params['it_b_pay'] . '</pay_expire>' : '') .
- '</direct_trade_create_req>';
- $request_data = $this->buildSignedParameters(array(
- 'service' => $this->service,
- 'partner' => $this->config['partner'],
- 'sec_id' => $this->config['sign_type'],
- 'format' => 'xml',
- 'v' => '2.0',
- 'req_id' => $params['req_id'],
- 'req_data' => $xml_str
- ));
- $url = $this->gateway;
- $input_charset = trim(strtolower($this->config['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->config['cacert']); //证书地址
- 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, $request_data); // post传输数据
- $responseText = curl_exec($curl);
- //var_dump( curl_error($curl) );//如果执行curl过程中出现异常,可打开此开关,以便查看异常内容
- curl_close($curl);
- if (empty($responseText))
- {
- return NULL;
- }
- parse_str($responseText, $responseData);
- if (empty($responseData['res_data']))
- {
- return NULL;
- }
- if ($this->config['sign_type'] == '0001')
- {
- $responseData['res_data'] = $this->rsaDecrypt($responseData['res_data'], $this->config['private_key_path']);
- }
- //token从res_data中解析出来(也就是说res_data中已经包含token的内容)
- $doc = new DOMDocument();
- $doc->loadXML($responseData['res_data']);
- $responseData['request_token'] = $doc->getElementsByTagName("request_token")->item(0)->nodeValue;
- $xml_str = '<auth_and_execute_req>' .
- '<request_token>' . $responseData['request_token'] . '</request_token>' .
- '</auth_and_execute_req>';
- return array(
- 'service' => self::SERVICE_WAP_AUTH,
- 'partner' => $this->config['partner'],
- 'sec_id' => $this->config['sign_type'],
- 'format' => 'xml',
- 'v' => '2.0',
- 'req_data' => $xml_str
- );
- }
- /**
- * 支付完成验证返回参数(包含同步和异步)
- *
- * @return <Boolean>
- */
- function verifyCallback()
- {
- $async = empty($_GET);
- $data = $async ? $_POST : $_GET;
- if (empty($data))
- {
- return FALSE;
- }
- $signValid = $this->verifyParameters($data, $data["sign"]);
- $notify_id = isset($data['notify_id']) ? $data['notify_id'] : NULL;
- if ($async && $this->is_mobile)
- {
- //对notify_data解密
- if ($this->config['sign_type'] == '0001')
- {
- $data['notify_data'] = $this->rsaDecrypt($data['notify_data'], $this->config['private_key_path']);
- }
- //notify_id从decrypt_post_para中解析出来(也就是说decrypt_post_para中已经包含notify_id的内容)
- $doc = new DOMDocument();
- $doc->loadXML($data['notify_data']);
- $notify_id = $doc->getElementsByTagName('notify_id')->item(0)->nodeValue;
- }
- //获取支付宝远程服务器ATN结果(验证是否是支付宝发来的消息)
- $responseTxt = 'true';
- if (!empty($notify_id))
- {
- $responseTxt = $this->verifyFromServer($notify_id);
- }
- //验证
- //$signValid的结果不是true,与安全校验码、请求时的参数格式(如:带自定义参数等)、编码格式有关
- //$responsetTxt的结果不是true,与服务器设置问题、合作身份者ID、notify_id一分钟失效有关
- return $signValid && preg_match("/true$/i", $responseTxt);
- }
- function verifyParameters($params, $sign)
- {
- $params = $this->filterSignParameter($params);
- if (isset($params['notify_data']))
- {
- $params = array(
- 'service' => $params['service'],
- 'v' => $params['v'],
- 'sec_id' => $params['sec_id'],
- 'notify_data' => $params['notify_data']
- );
- }
- else
- {
- ksort($params);
- reset($params);
- }
- $content = urldecode(http_build_query($params));
- switch (strtoupper(trim($this->config['sign_type'])))
- {
- case "MD5" :
- return md5($content . $this->config['key']) == $sign;
- case "RSA" :
- case "0001" :
- return $this->rsaVerify($content, $this->config['public_key_path'], $sign);
- default :
- return FALSE;
- }
- }
- /**
- * 过滤参数,去除sign/sign_type参数
- * @param $params
- * @return <Array>
- */
- function filterSignParameter($params)
- {
- $result = array();
- foreach ($params as $key => $value)
- {
- if ($key != 'sign' && $key != 'sign_type' && $value)
- {
- $result[$key] = $value;
- }
- }
- return $result;
- }
- function verifyFromServer($notify_id)
- {
- $transport = strtolower(trim($this->config['transport']));
- $partner = trim($this->config['partner']);
- $veryfy_url = ($transport == 'https' ? self::VERIFY_URL_HTTPS : self::VERIFY_URL) . "partner=$partner¬ify_id=$notify_id";
- $curl = curl_init($veryfy_url);
- curl_setopt($curl, CURLOPT_HEADER, 0); // 过滤HTTP头
- curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true); //SSL证书认证
- curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2); //严格认证
- curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1); // 显示输出结果
- curl_setopt($curl, CURLOPT_CAINFO, $this->config['cacert']); //证书地址
- $responseText = curl_exec($curl);
- // var_dump( curl_error($curl) );//如果执行curl过程中出现异常,可打开此开关,以便查看异常内容
- curl_close($curl);
- return $responseText;
- }
- /**
- * RSA验签,注意验签的公钥是支付宝的公钥,不是自己生成的rsa公钥,可以在淘宝的demo中获得
- * @param $data string 待签名数据
- * @param $ali_public_key_path string 支付宝的公钥文件路径
- * @param $sign string 要校对的的签名结果
- * @return <Boolean> 验证结果
- * @throws Exception
- */
- function rsaVerify($data, $ali_public_key_path, $sign)
- {
- $pubKey = file_get_contents($ali_public_key_path);
- $res = openssl_get_publickey($pubKey);
- if (!$res)
- {
- throw new Exception('公钥格式错误');
- }
- $result = (bool) openssl_verify($data, base64_decode($sign), $res);
- openssl_free_key($res);
- return $result;
- }
- /**
- * RSA解密
- * @param $content string 需要解密的内容,密文
- * @param $private_key_path string 商户私钥文件路径
- * @return string 解密后内容,明文
- */
- function rsaDecrypt($content, $private_key_path)
- {
- $priKey = file_get_contents($private_key_path);
- $res = openssl_get_privatekey($priKey);
- //用base64将内容还原成二进制
- $content = base64_decode($content);
- //把需要解密的内容,按128位拆开解密
- $result = '';
- for ($i = 0; $i < strlen($content) / 128; $i++)
- {
- $data = substr($content, $i * 128, 128);
- openssl_private_decrypt($data, $decrypt, $res);
- $result .= $decrypt;
- }
- openssl_free_key($res);
- return $result;
- }
- }
|