123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449 |
- <?php
- namespace Lib\Payment;
- /**
- * 支付宝接口类
- */
- class Alipay{
- /**
- *支付宝网关地址(新)
- */
- private $alipay_gateway_new = 'https://mapi.alipay.com/gateway.do?';
- /**
- * 消息验证地址
- *
- * @var string
- */
- private $alipay_verify_url = 'http://notify.alipay.com/trade/notify_query.do?';
- /**
- * 支付接口标识
- *
- * @var string
- */
- private $code = 'alipay';
- /**
- * 支付接口配置信息
- *
- * @var array
- */
- private $payment;
- /**
- * 订单信息
- *
- * @var array
- */
- private $order;
- /**
- * 发送至支付宝的参数
- *
- * @var array
- */
- private $parameter;
- /**
- * 订单类型 product_buy商品购买,predeposit预存款充值
- * @var unknown
- */
- private $order_type;
-
- public function __construct($payment_info=array(),$order_info=array()){
- // if (!extension_loaded('openssl')) $this->alipay_verify_url = 'http://notify.alipay.com/trade/notify_query.do?';
- if(!empty($payment_info) and !empty($order_info)){
- $this->payment = $payment_info;
- $this->order = $order_info;
- }else{
- $this->payment = $payment_info;
-
- }
- }
- /**
- * 获取支付接口的请求地址
- *
- * @return string
- */
- public function get_payurl(){
- $this->parameter = array(
- 'service' => $this->payment['alipay_service'], //服务名
- 'partner' => $this->payment['alipay_partner'], //合作伙伴ID
- 'key' => $this->payment['alipay_key'],
- '_input_charset' => strtolower('utf-8'), //网站编码
- 'notify_url' => $this->order['notify_url'], //通知URL
-
- 'sign_type' => 'MD5', //签名方式
- 'return_url' => $this->order['return_url'], //返回URL
-
- 'subject' => $this->order['subject'], //商品名称
- 'out_trade_no' => $this->order['pay_order_no'], //外部交易编号
- 'payment_type' => 1, //支付类型
- 'logistics_type' => 'EXPRESS', //物流配送方式:POST(平邮)、EMS(EMS)、EXPRESS(其他快递)
- 'logistics_payment' => 'BUYER_PAY', //物流费用付款方式:SELLER_PAY(卖家支付)、BUYER_PAY(买家支付)、BUYER_PAY_AFTER_RECEIVE(货到付款)
- 'logistics_fee' => '0.00',
- 'receive_name' => $this->order['name'],//收货人姓名
- 'seller_email' => $this->payment['alipay_account'], //卖家邮箱
- 'price' => $this->order['pay_total'],//订单总价
- 'quantity' => 1,//商品数量
- );
-
- $this->parameter['sign'] = $this->sign($this->parameter);
-
- return $this->create_url();
- }
- /**
- * 通知地址验证
- *
- * @return bool
- */
- public function notify_verify() {
- $param = $_POST;
- $param['key'] = $this->payment['alipay_key'];
- $veryfy_url = $this->alipay_verify_url. "partner=" .$this->payment['alipay_partner']. "¬ify_id=".$param["notify_id"];
- $veryfy_result = $this->getHttpResponse($veryfy_url);
- $mysign = $this->sign($param);
- if (preg_match("/true$/i",$veryfy_result) && $mysign == $param["sign"]) {
- // $this->order_type = $param['extra_common_param'];
- return true;
- } else {
- return false;
- }
- }
- /**
- * 返回地址验证
- *
- * @return bool
- */
- public function return_verify() {
- //生成签名结果
- $isSign = $this->getSignVeryfy($_GET, $_GET["sign"]);
-
- //获取支付宝远程服务器ATN结果(验证是否是支付宝发来的消息)
- $responseTxt = 'true';
- if (! empty($_GET["notify_id"])) {
- $responseTxt = $this->getResponse($_GET["notify_id"]);
- }
- if (preg_match("/true$/i",$responseTxt) && $isSign) {
- return true;
- } else {
- return false;
- }
-
- }
- /**
- * 除去数组中的空值和签名参数
- * @param $para 签名参数组
- * return 去掉空值与签名参数后的新签名参数组
- */
- function paraFilter($para) {
- $para_filter = array();
- while (list ($key, $val) = each ($para)) {
- if($key == "sign" || $key == "sign_type" || $val == "")continue;
- else $para_filter[$key] = $para[$key];
- }
- return $para_filter;
- }
- /**
- * 对数组排序
- * @param $para 排序前的数组
- * return 排序后的数组
- */
- function argSort($para) {
- ksort($para);
- reset($para);
- return $para;
- }
- /**
- * 把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串
- * @param $para 需要拼接的数组
- * return 拼接完成以后的字符串
- */
- function createLinkstring($para) {
- $arg = "";
- while (list ($key, $val) = each ($para)) {
- $arg.=$key."=".$val."&";
- }
- //去掉最后一个&字符
- $arg = substr($arg,0,count($arg)-2);
-
- //如果存在转义字符,那么去掉转义
- if(get_magic_quotes_gpc()){$arg = stripslashes($arg);}
-
- return $arg;
- }
- /**
- * 签名字符串
- * @param $prestr 需要签名的字符串
- * @param $key 私钥
- * return 签名结果
- */
- function md5Sign($prestr, $key) {
- $prestr = $prestr . $key;
- return md5($prestr);
- }
- /**
- * 验证签名
- * @param $prestr 需要签名的字符串
- * @param $sign 签名结果
- * @param $key 私钥
- * return 签名结果
- */
- function md5Verify($prestr, $sign, $key) {
- $prestr = $prestr . $key;
- $mysgin = md5($prestr);
-
- if($mysgin == $sign) {
- return true;
- }
- else {
- return false;
- }
- }
- function getHttpResponseGET($url,$cacert_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,$cacert_url);//证书地址
- $responseText = curl_exec($curl);
- curl_close($curl);
-
- return $responseText;
- }
- /**
- * 获取远程服务器ATN结果,验证返回URL
- * @param $notify_id 通知校验ID
- * @return 服务器ATN结果
- * 验证结果集:
- * invalid命令参数不对 出现这个错误,请检测返回处理中partner和key是否为空
- * true 返回正确信息
- * false 请检查防火墙或者是服务器阻止端口问题以及验证时间是否超过一分钟
- */
- function getResponse($notify_id) {
- $transport = strtolower(trim('http'));
- $partner = trim($this->payment['alipay_partner']);
- $veryfy_url = '';
- if($transport == 'https') {
- $veryfy_url = $this->https_verify_url;
- }
- else {
- $veryfy_url = $this->alipay_verify_url;
- }
- $veryfy_url = $this->alipay_verify_url."partner=" . $partner . "¬ify_id=" . $notify_id;
- $cacert=ROOT_PATH.'Public/pay/cacert.pem';
-
- $responseTxt = $this->getHttpResponseGET($veryfy_url, $cacert);
-
- return $responseTxt;
- }
- function getSignVeryfy($para_temp, $sign) {
- //除去待签名参数数组中的空值和签名参数
- $para_filter = $this->paraFilter($para_temp);
-
- //对待签名参数数组排序
- $para_sort = $this->argSort($para_filter);
-
- //把数组所有元素,按照“参数=参数值”的模式用“&”字符拼接成字符串
- $prestr = $this->createLinkstring($para_sort);
-
- $isSgin = false;
- switch (strtoupper(trim(strtoupper('MD5')))) {
- case "MD5" :
- $isSgin = $this->md5Verify($prestr, $sign, $this->payment['alipay_key']);
-
- break;
- default :
- $isSgin = false;
- }
-
- return $isSgin;
- }
- /**
- * 针对notify_url验证消息是否是支付宝发出的合法消息
- * @return 验证结果
- */
- function verifyNotify(){
- if(empty($_POST)) {//判断POST来的数组是否为空
- return false;
- }
- else {
- //生成签名结果
- $isSign = $this->getSignVeryfy($_POST, $_POST["sign"]);
- //获取支付宝远程服务器ATN结果(验证是否是支付宝发来的消息)
- $responseTxt = 'false';
- if (! empty($_POST["notify_id"])) {$responseTxt = $this->getResponse($_POST["notify_id"]);}
-
- //验证
- //$responsetTxt的结果不是true,与服务器设置问题、合作身份者ID、notify_id一分钟失效有关
- //isSign的结果不是true,与安全校验码、请求时的参数格式(如:带自定义参数等)、编码格式有关
- if (preg_match("/true$/i",$responseTxt) && $isSign) {
- return true;
- } else {
- return false;
- }
- }
- }
-
-
- /**
- *
- * 取得订单支付状态,成功或失败
- * @param array $param
- * @return array
- */
- public function getPayResult($param){
- return $param['trade_status'] == 'TRADE_SUCCESS';
- }
- /**
- *
- *
- * @param string $name
- * @return
- */
- public function __get($name){
- return $this->$name;
- }
- /**
- * 远程获取数据
- * $url 指定URL完整路径地址
- * @param $time_out 超时时间。默认值:60
- * return 远程输出的数据
- */
- private function getHttpResponse($url,$time_out = "60") {
- $urlarr = parse_url($url);
- $errno = "";
- $errstr = "";
- $transports = "http";
- $responseText = "";
- if($urlarr["scheme"] == "https") {
- $transports = "ssl://";
- $urlarr["port"] = "443";
- } else {
- $transports = "tcp://";
- $urlarr["port"] = "80";
- }
- $fp=@fsockopen($transports . $urlarr['host'],$urlarr['port'],$errno,$errstr,$time_out);
- if(!$fp) {
- die("ERROR: $errno - $errstr<br />\n");
- } else {
- if (trim(CHARSET) == '') {
- fputs($fp, "POST ".$urlarr["path"]." HTTP/1.1\r\n");
- } else {
- fputs($fp, "POST ".$urlarr["path"].'?_input_charset='.CHARSET." HTTP/1.1\r\n");
- }
- fputs($fp, "Host: ".$urlarr["host"]."\r\n");
- fputs($fp, "Content-type: application/x-www-form-urlencoded\r\n");
- fputs($fp, "Content-length: ".strlen($urlarr["query"])."\r\n");
- fputs($fp, "Connection: close\r\n\r\n");
- fputs($fp, $urlarr["query"] . "\r\n\r\n");
- while(!feof($fp)) {
- $responseText .= @fgets($fp, 1024);
- }
- fclose($fp);
- $responseText = trim(stristr($responseText,"\r\n\r\n"),"\r\n");
- return $responseText;
- }
- }
- /**
- * 制作支付接口的请求地址
- *
- * @return string
- */
- private function create_url() {
- $url = $this->alipay_gateway_new;
- $filtered_array = $this->para_filter($this->parameter);
- $sort_array = $this->arg_sort($filtered_array);
- $arg = "";
- while (list ($key, $val) = each ($sort_array)) {
- $arg.=$key."=".urlencode($val)."&";
- }
- $url.= $arg."sign=" .$this->parameter['sign'] ."&sign_type=".$this->parameter['sign_type'];
-
- return $url;
- }
- /**
- * 取得支付宝签名
- *
- * @return string
- */
- private function sign($parameter) {
- $mysign = "";
-
- $filtered_array = $this->para_filter($parameter);
- $sort_array = $this->arg_sort($filtered_array);
- $arg = "";
- while (list ($key, $val) = each ($sort_array)) {
- $arg .= $key."=".$this->charset_encode($val,(empty($parameter['_input_charset'])?"UTF-8":$parameter['_input_charset']),(empty($parameter['_input_charset'])?"UTF-8":$parameter['_input_charset']))."&";
- }
- $prestr = substr($arg,0,-1); //去掉最后一个&号
- $prestr .= $parameter['key'];
- if($parameter['sign_type'] == 'MD5') {
- $mysign = md5($prestr);
- }elseif($parameter['sign_type'] =='DSA') {
- //DSA 签名方法待后续开发
- die("DSA 签名方法待后续开发,请先使用MD5签名方式");
- }else {
- die("支付宝暂不支持".$parameter['sign_type']."类型的签名方式");
- }
- return $mysign;
- }
- /**
- * 除去数组中的空值和签名模式
- *
- * @param array $parameter
- * @return array
- */
- private function para_filter($parameter) {
- $para = array();
- while (list ($key, $val) = each ($parameter)) {
- if($key == "sign" || $key == "sign_type" || $key == "key" || $val == "")continue;
- else $para[$key] = $parameter[$key];
- }
- return $para;
- }
- /**
- * 重新排序参数数组
- *
- * @param array $array
- * @return array
- */
- private function arg_sort($array) {
- ksort($array);
- reset($array);
- return $array;
- }
- /**
- * 实现多种字符编码方式
- */
- private function charset_encode($input,$_output_charset,$_input_charset="UTF-8") {
- $output = "";
- if(!isset($_output_charset))$_output_charset = $this->parameter['_input_charset'];
- if($_input_charset == $_output_charset || $input == null) {
- $output = $input;
- } elseif (function_exists("mb_convert_encoding")){
- $output = mb_convert_encoding($input,$_output_charset,$_input_charset);
- } elseif(function_exists("iconv")) {
- $output = iconv($_input_charset,$_output_charset,$input);
- } else die("sorry, you have no libs support for charset change.");
- return $output;
- }
- }
|