当前位置:首页 > PHP教程 > php高级应用 > 列表

php版本的支付宝服务窗API接口的开发

发布:smiling 来源: PHP粉丝网  添加日期:2016-07-15 16:11:32 浏览: 评论:0 

支付宝服务窗API接口的开发对于许多网站要充值的朋友来讲是非常的重要的,今天我们就一起来看一篇关于php版本的支付宝服务窗API接口的开发例子。

这两天没事要接入支付宝服务窗,看支付宝的DEMO,我的神,我怎么评价好呢?开发思路很牛逼,但是阅读性不是很好,很阻碍简单的开发。所以我就根据提供的API简单的开发了点,接口还有很多不完善,有兴趣的可以自己完善一下,下边我就把代码贴出来,有时间再写如何使用。

  1. <?php 
  2.  
  3. class AlipayService{ 
  4.     /** 
  5.      - 服务接口信息 
  6.      */ 
  7.     public $service = null; 
  8.     /** 
  9.      - 签名信息 
  10.      */ 
  11.     public $sign = null; 
  12.     /** 
  13.      - 签名类型 
  14.      */ 
  15.     public $sign_type = null; 
  16.     /** 
  17.      - 字符集 
  18.      */ 
  19.     public $charset = null; 
  20.     /** 
  21.      - 解析的biz_content数据 
  22.      */ 
  23.     public $request = null; 
  24.     /** 
  25.      - 用户openid 
  26.      */ 
  27.     public $from_user_id = null; 
  28.     /** 
  29.      - 消息类型 
  30.      */ 
  31.     public $msg_type = null; 
  32.     /** 
  33.      - 事件类型 
  34.      */ 
  35.     public $event_type = null; 
  36.     /** 
  37.      - 行为参数 
  38.      */ 
  39.     public $action_param = null; 
  40.     /** 
  41.      - 支付宝用户信息 
  42.      */ 
  43.     public $user_info = null; 
  44.     /** 
  45.      - 文本消息内容 
  46.      */ 
  47.     public $text = null; 
  48.     /** 
  49.      - 图片媒体id 
  50.      */ 
  51.     public $media_id = null; 
  52.     /** 
  53.      - 图片格式 
  54.      */ 
  55.     public $format = null; 
  56.     /** 
  57.      - 是否开启调试 
  58.      */ 
  59.     private $debug = false; 
  60.     /** 
  61.      - 接口类型 
  62.      */ 
  63.     private $interface_type = array(     
  64.             'qrcode'      => 'alipay.mobile.public.qrcode.create',   
  65.             'follow'      => 'alipay.mobile.public.follow.list',     
  66.             'gis_get'     => 'alipay.mobile.public.gis.get',     
  67.             'menu_get'    => 'alipay.mobile.public.menu.get',          
  68.             'menu_add'    => 'alipay.mobile.public.menu.add'
  69.             'down_media'  => 'alipay.mobile.public.multimedia.download'
  70.             'menu_update' => 'alipay.mobile.public.menu.update',     
  71.             'info_query'  => 'alipay.mobile.public.info.query',  
  72.             'info_modify' => 'alipay.mobile.public.info.modify'
  73.             'shortlink'   => 'alipay.mobile.public.shortlink.create',    
  74.             'label_add'   => 'alipay.mobile.public.label.add',   
  75.             'label_del'   => 'alipay.mobile.public.label.delete',    
  76.             'label_update'        => 'alipay.mobile.public.label.update',    
  77.             'label_query'         => 'alipay.mobile.public.label.query',     
  78.             'label_user_add'      => 'alipay.mobile.public.label.user.add',  
  79.             'label_user_del'      => 'alipay.mobile.public.label.user.delete',   
  80.             'label_user_query'    => 'alipay.mobile.public.label.user.query',    
  81.             'message_custom'      => 'alipay.mobile.public.message.custom.send',     
  82.             'message_total'       => 'alipay.mobile.public.message.total.send',  
  83.             'message_single'      => 'alipay.mobile.public.message.single.send',     
  84.             'message_label_send'  => 'alipay.mobile.public.message.label.send',  
  85.         ); 
  86.     /** 
  87.      - 私有密钥地址,替换为你自己的 
  88.      */ 
  89.     private $private_rsa_key_path ='rsa_private_key.pem'
  90.     /** 
  91.      - 私有密钥地址,替换为你自己的 
  92.      */ 
  93.     private $public_rsa_key_path ='rsa_public_key.pem'
  94.     /** 
  95.      - 支付宝窗的app id 替换成你自己的 
  96.      */ 
  97.     private $app_id = '2015120200901652'
  98.     /** 
  99.      - 开启DEBUG参数 
  100.      - @params bool  debug  true 开启调试 false 关闭调试 
  101.      - @author widuu <admin@widuu.com> 
  102.      */ 
  103.     public function __construct( $debug = false ){ 
  104.         /* 是否开启DEBUG */ 
  105.         if$debug ) $this->debug = true; 
  106.     } 
  107.     /** 
  108.      - 获取参数,解析请求参数 
  109.      -  
  110.      - @author widuu <admin@widuu.com> 
  111.      */ 
  112.     public function get_request(){ 
  113.         if( !emptyempty($_POST) ){ 
  114.             // 请求的服务接口 
  115.             $this->service = $_POST['service']; 
  116.             // 获取请求字符集 
  117.             $this->charset = $_POST['charset']; 
  118.             // 获取请求的biz_content 
  119.             $request_biz_content = $_POST['biz_content']; 
  120.             // 加密算法 
  121.             $this->sign_type = $_POST['sign_type']; 
  122.             // 加密字符串 
  123.             $this->sign = $_POST['sign']; 
  124.             // 如果请求格式不是Utf-8 转换格式为Utf-8 
  125.             ifstrtolower($this->charset) != 'utf-8' ){ 
  126.                 $request_biz_content = iconv('GBK''utf-8'$request_biz_content); 
  127.             } 
  128.             // 解析字符串为xml 
  129.             $request_xml   = @simplexml_load_string($request_biz_content"SimpleXMLElement" , LIBXML_NOCDATA ); 
  130.             // 解析为数组 
  131.             $request_array = json_decode(json_encode($request_xml),true); 
  132.             $this->request = $request_array
  133.             /* 解析 */ 
  134.             $this->analysis($request_array); 
  135.             if($this->debug) $this->write_log('REQUEST_INFO',var_export($request_array,true)); 
  136.             // 默认验证方法 
  137.             if$this->service == 'alipay.service.check'){ 
  138.                 $this->verify($_POST); 
  139.                 exit(); 
  140.             } 
  141.             /* 返回结果 */ 
  142.             return $request_array
  143.         } 
  144.     } 
  145.     /** 
  146.      - 回复文本内容 
  147.      - @params string content  文本数据 
  148.      - @params bool   mass     ture为群发 
  149.      - @author widuu <admin@widuu.com> 
  150.      */ 
  151.     public function text($content,$mass=false){ 
  152.         $info['text'] = array'content' => $content ); 
  153.         /* 组织内容 */ 
  154.         $biz_content = $this->common_response('text',$info,$mass); 
  155.         /* 判断是否为群发 */ 
  156.         if($mass){ 
  157.             $method = 'message_total'
  158.         }else
  159.             $method = 'message_custom'
  160.         } 
  161.         $sys_params = $this->common_system($method,$biz_content); 
  162.         $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params)); 
  163.         /* 返回结果 结果是JSON数据 */ 
  164.         $result = $this->response_post($sys_params); 
  165.         return $result
  166.     } 
  167.     /** 
  168.      - 回复图文内容 
  169.      - @params array articles  拼接的图文消息数组 
  170.      - @params bool   mass     ture为群发 
  171.      - @author widuu <admin@widuu.com> 
  172.      */ 
  173.     public function articles($articles,$mass=false){ 
  174.         $info['articles'] = array($articles); 
  175.         /* 组织内容 */ 
  176.         $biz_content = $this->common_response('image-text',$info,$mass); 
  177.         /* 判断是否群发 */ 
  178.         if($mass){ 
  179.             $method = 'message_total'
  180.         }else
  181.             $method = 'message_custom'
  182.         } 
  183.         /* 加密参数 */ 
  184.         $sys_params = $this->common_system($method,$biz_content); 
  185.         /* 加密字符 */ 
  186.         $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params)); 
  187.         /* 返回结果 结果是JSON数据 */ 
  188.         $result = $this->response_post($sys_params); 
  189.         return $result
  190.     } 
  191.     /** 
  192.      - 关注事件 
  193.      -  
  194.      - @author widuu <admin@widuu.com> 
  195.      */ 
  196.     public function is_follow(){ 
  197.         $request = $this->request; 
  198.         if$request['MsgType'] == 'event' && $request['EventType'] == 'follow' ){ 
  199.             return true; 
  200.         }else
  201.             return false; 
  202.         } 
  203.     } 
  204.     /** 
  205.      - 取消关注事件 
  206.      -  
  207.      - @author widuu <admin@widuu.com> 
  208.      */ 
  209.     public function is_unfollow(){ 
  210.         $request = $this->request; 
  211.         if$request['MsgType'] == 'event' && $request['EventType'] == 'unfollow' ){ 
  212.             return true; 
  213.         }else
  214.             return false; 
  215.         } 
  216.     } 
  217.     /** 
  218.      - 下载用户发来的图片 
  219.      - @param  media_id string  图片id 
  220.      - @param  filename string  保存图片地址和名称 
  221.      - @author widuu <admin@widuu.com> 
  222.      */ 
  223.     public function down_media($media_id,$filename){ 
  224.         $sys_params = $this->common_system('down_media',array('mediaId'=>$media_id)); 
  225.         $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params)); 
  226.         /* 返回数据 */ 
  227.         $result = $this->response_post($sys_params,true); 
  228.         $result = file_put_contents($filename$result); 
  229.         if$this->debug ){ 
  230.             $this->write_log('SAVE_IMAGE','保存图片'.(string)$result); 
  231.         } 
  232.         return $result
  233.     } 
  234.     /** 
  235.      - (添加|更新|获取)自定义菜单 
  236.      - @param  string $type  (add|update|get) 
  237.      - @param  array  $menu   菜单数组,如果是获取菜单可以留空 
  238.      - @author widuu <admin@widuu.com> 
  239.      */ 
  240.     public function menu( $type,$menu = array() ){ 
  241.         if( !in_array( $typearray('get','update','add')) ){ 
  242.             if$this->debug ){ 
  243.                 $this->write_log('ERROR','菜单操作方法错误'); 
  244.             } 
  245.             return false; 
  246.         } 
  247.         /* 拼接接口方法 */ 
  248.         $method = 'menu_'.$type
  249.         $sys_params = $this->common_system($method,$menu); 
  250.         /* 加密字符串 */ 
  251.         $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params)); 
  252.         /* 请求获取结果 */ 
  253.         $result = $this->response_post($sys_params); 
  254.         /* 转义并解析JSON 数据 */ 
  255.         $menu_json = json_decode(iconv('GBK''utf-8'$result),true); 
  256.         /* 组织接口信息 */ 
  257.         $interface = 'alipay_mobile_public_'.$method.'_response'
  258.         /* 遇到错误返回 */ 
  259.         if$menu_json[$interface]['code'] != 200 ){ 
  260.             if$this->debug ){ 
  261.                 $this->write_log('GET_MENU_ERROR',$menu_json[$interface]['msg']); 
  262.             } 
  263.             return false; 
  264.         } 
  265.         /* 根据类型不同返回不同的结果 */ 
  266.         if$type == 'get' ){ 
  267.             return $menu_json[$interface]['menu_content']; 
  268.         }else
  269.             return $menu_json[$interface]['msg']; 
  270.         } 
  271.     } 
  272.  
  273.     /** 
  274.      - POST数据方法 
  275.      - @param  array params 参数数组 
  276.      - @author widuu <admin@widuu.com> 
  277.      */ 
  278.     private function response_post($params,$type=false){ 
  279.         // 下载媒体和请求网关 
  280.         if($down){ 
  281.             $url = 'https://openfile.alipay.com/chat/multimedia.do'
  282.         }else
  283.             $url = 'https://openapi.alipay.com/gateway.do'
  284.         } 
  285.         $ch = curl_init(); 
  286.         curl_setopt($ch, CURLOPT_URL, $url); 
  287.         curl_setopt($ch, CURLOPT_HEADER, 0); 
  288.         curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); 
  289.         curl_setopt($ch, CURLOPT_POST, 1); 
  290.         curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($params)); 
  291.         curl_setopt($ch, CURLOPT_FOLLOWLOCATION,true); 
  292.         curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); 
  293.         curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false); 
  294.         curl_setopt($ch, CURLOPT_SSLVERSION, CURL_SSLVERSION_TLSv1); 
  295.         $curl = curl_exec($ch); 
  296.         curl_close($ch); 
  297.         return $curl
  298.     } 
  299.     /** 
  300.      - 拼接回复数据 
  301.      - @param   string $type  回复类型 
  302.      - @param   array  $info  回复内容 
  303.      - @param   bool   $mass  是否为群发 
  304.      - @author widuu <admin@widuu.com> 
  305.      */ 
  306.     private function common_response($type,$info,$mass=false){ 
  307.         $request = $this->request; 
  308.         $params = array(); 
  309.         // 如果不是群发 
  310.         if( !$mass ) $params['toUserId'] = $request['FromUserId']; 
  311.         $params['msgType'] = $type
  312.         $params['createTime'] = time(); 
  313.         $content = array_merge($params,$info); 
  314.         return $content
  315.     } 
  316.     /** 
  317.      - 拼接加密参数 
  318.      - @param   string $interface_type  接口类型 
  319.      - @param   array  $biz_content     返回biz_content的数组 
  320.      - @author widuu <admin@widuu.com> 
  321.      */ 
  322.  
  323.     private function common_system($interface_type,$biz_content){ 
  324.         /* 接口集合 */ 
  325.         $type = $this->interface_type; 
  326.         $method = $type[$interface_type]; 
  327.         /* 公共参数 */ 
  328.         $params = array ( 
  329.             'method' => $method
  330.             'charset' => 'UTF-8'
  331.             'sign_type' => 'RSA'
  332.             'app_id' => $this->app_id, 
  333.             'timestamp' => date ( 'Y-m-d H:i:s', time () ), 
  334.             'version'=>'1.0'
  335.         ); 
  336.         /* 获取某些接口时没有biz_content参数 */ 
  337.         ifcount($biz_content) > 0 ){ 
  338.             $params['biz_content'] = json_encode($biz_content); 
  339.         } 
  340.         /* 返回系统参数 */ 
  341.         return $params
  342.     } 
  343.     /** 
  344.      - 服务验证 
  345.      - @params array params  是自动获的验证信息 
  346.      - @author widuu <admin@widuu.com> 
  347.      */ 
  348.     private function verify($params){ 
  349.         /* 参数为空 */ 
  350.         ifemptyempty($params) ){ 
  351.             if$this->debug ){ 
  352.                 $this->write_log('ERROR','验证参数为空'); 
  353.             } 
  354.         } 
  355.         /* 构建参数,使用字典排序再拼接字符串 */ 
  356.         $query_data = $this->build_query($params); 
  357.         /* 验证信息,有可能php版本BUG不支持验证 */ 
  358.         $verify_result = $this->ras_verify($query_data); 
  359.         /* 返回验证结果 */ 
  360.         if$verify_result ){ 
  361.             /* 取公有密钥的字符串合并为一行 */ 
  362.             $public_rsa_string = file_get_contents($this->public_rsa_key_path); 
  363.             $public_rsa_string = str_replace ( "-----BEGIN PUBLIC KEY-----"""$public_rsa_string ); 
  364.             $public_rsa_string = str_replace ( "-----END PUBLIC KEY-----"""$public_rsa_string ); 
  365.             $public_rsa_string = str_replace ( "\r"""$public_rsa_string ); 
  366.             $public_rsa_string = str_replace ( "\n"""$public_rsa_string ); 
  367.             /* 构建加密字符串 */ 
  368.             $response_xml = "<success>true</success><biz_content>$public_rsa_string</biz_content>"
  369.             /* 生成验证信息 */ 
  370.             $sign = $this->rsa_sign (  $response_xml ); 
  371.             /* 构建返回数据 */ 
  372.             $response = "<?xml version=\"1.0\" encoding=\"GBK\"?><alipay><response>$response_xml</response><sign>$sign</sign><sign_type>RSA</sign_type></alipay>"
  373.             if$this->debug ){ 
  374.                 $this->write_log('CHECK_RESPONSE',$response); 
  375.             } 
  376.             /* 输出返回信息 */ 
  377.             echo $response
  378.             exit(); 
  379.         }else
  380.             if$this->debug ){ 
  381.                 $this->write_log('ERROR','验证失败'); 
  382.             } 
  383.         } 
  384.     } 
  385.     /** 
  386.      - 拼接为字符串函数 
  387.      - @params array  params  拼接函数 
  388.      - @author widuu <admin@widuu.com> 
  389.      */ 
  390.     private function build_query($params){ 
  391.         /* 删除sign字符串 */ 
  392.         unset($params['sign']); 
  393.         /* 字典排序 */ 
  394.         ksort($params); 
  395.         /* 拼接 */  
  396.         $query_array = array(); 
  397.         foreach ($params as $k => $v) { 
  398.             $query_array[] = "$k"."="."$v"
  399.         } 
  400.         $query_data = implode("&"$query_array); 
  401.         /* 返回拼接好的字符串 */ 
  402.         return $query_data
  403.     } 
  404.     /** 
  405.      - 验证加密sign,有些PHP版本不支持,不支持情况直接返回true 
  406.      - @params string query_data  加密字符串 
  407.      - @author widuu <admin@widuu.com> 
  408.      */ 
  409.     private function ras_verify($query_data){ 
  410.         /* 读取公钥文件,PEM格式 */ 
  411.         $pubKey = file_get_contents($this->public_rsa_key_path); 
  412.         /* 转换为openssl格式密钥 */ 
  413.         $res = openssl_get_publickey($pubKey); 
  414.         /* 调用openssl内置方法验签 */ 
  415.         $result = (bool) openssl_verify($query_database64_decode($this->sign), $res); 
  416.         /* 释放资源 */ 
  417.         openssl_free_key($res); 
  418.         /* 有些PHP版本错误,直接返回true */ 
  419.         ifstrpos( openssl_error_string(),'PEM_read_bio' ) ){   
  420.              return true; 
  421.         } 
  422.         /* 返回验签结果 */ 
  423.         return $result
  424.     } 
  425.     /** 
  426.      - 通过私有密钥加密数据 
  427.      - @params string data  加密数据 
  428.      - @author widuu <admin@widuu.com> 
  429.      */ 
  430.     private function rsa_sign($data) { 
  431.         /* 读取私钥 */ 
  432.         $priKey = file_get_contents ( $this->private_rsa_key_path ); 
  433.         /* 转换为openssl格式密钥 */ 
  434.         $res = openssl_get_privatekey ( $priKey ); 
  435.         /* 调用openssl 加密 */ 
  436.         openssl_sign ( $data$sign$res ); 
  437.         /* 释放资源 */ 
  438.         openssl_free_key ( $res ); 
  439.         /* Base64加密 */ 
  440.         $sign = base64_encode ( $sign ); 
  441.         /* 返回加密参数 */ 
  442.         return $sign
  443.     } 
  444.     private function analysis($params){ 
  445.         switch($params['MsgType']){ 
  446.             case 'image'
  447.                 $this->media_id = $params['Image']['MediaId']; 
  448.                 $this->format   = $params['Image']['Format'];  
  449.                 break
  450.             case 'text'
  451.                 $this->text = $params['Text']['Content']; 
  452.                 break
  453.             case 'event'
  454.                 $this->event_type   = $params['EventType']; 
  455.                 $this->action_param = $params['ActionParam']; 
  456.                 break
  457.             default
  458.                 break
  459.         } 
  460.         $this->msg_type  = $params['MsgType']; 
  461.         $this->user_info = json_decode($params['UserInfo'],true); 
  462.     } 
  463.     /** 
  464.      - DEBUG 为true时的拼接字符串 
  465.      - @param   string  $level    自定义标识符 
  466.      - @param   string  $info     自定义内容 
  467.      - @param   string  $log_path 自定义日志路径 
  468.      - @author widuu <admin@widuu.com> 
  469.      */ 
  470.     public function write_log($level,$info,$log_path = '' ){ 
  471.         ifemptyempty($log_path) ){  //phpfensi.com 
  472.             $log_path = dirname ( __FILE__ ) . "/log.txt"
  473.         } 
  474.         file_put_contents($log_path"[$level]".date ( "Y-m-d H:i:s" ) . "  " . $info . "\r\n", FILE_APPEND ); 
  475.     } 
好了以上就是小编为各位整理的一篇关于支付宝服务窗API接口的开发例子,这个有前提条件的就是我们必须要申请一个权限才可以,这个官方可以申请小编就不介绍。

Tags: 服务窗 php接口 php支付宝

分享到: