热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

PHP支付宝服务窗API接口的开发

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

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


 'alipay.mobile.public.qrcode.create',  
            'follow'      => 'alipay.mobile.public.follow.list',    
            'gis_get'     => 'alipay.mobile.public.gis.get',    
            'menu_get'    => 'alipay.mobile.public.menu.get',         
            'menu_add'    => 'alipay.mobile.public.menu.add',
            'down_media'  => 'alipay.mobile.public.multimedia.download',
            'menu_update' => 'alipay.mobile.public.menu.update',    
            'info_query'  => 'alipay.mobile.public.info.query', 
            'info_modify' => 'alipay.mobile.public.info.modify',
            'shortlink'   => 'alipay.mobile.public.shortlink.create',   
            'label_add'   => 'alipay.mobile.public.label.add',  
            'label_del'   => 'alipay.mobile.public.label.delete',   
            'label_update'        => 'alipay.mobile.public.label.update',   
            'label_query'         => 'alipay.mobile.public.label.query',    
            'label_user_add'      => 'alipay.mobile.public.label.user.add', 
            'label_user_del'      => 'alipay.mobile.public.label.user.delete',  
            'label_user_query'    => 'alipay.mobile.public.label.user.query',   
            'message_custom'      => 'alipay.mobile.public.message.custom.send',    
            'message_total'       => 'alipay.mobile.public.message.total.send', 
            'message_single'      => 'alipay.mobile.public.message.single.send',    
            'message_label_send'  => 'alipay.mobile.public.message.label.send', 
        );

    /**
     - 私有密钥地址,替换为你自己的
     */

    private $private_rsa_key_path ='rsa_private_key.pem';

    /**
     - 私有密钥地址,替换为你自己的
     */

    private $public_rsa_key_path ='rsa_public_key.pem';

    /**
     - 支付宝窗的app id 替换成你自己的
     */

    private $app_id = '2015120200901652';

    /**
     - 开启DEBUG参数
     - @params bool  debug  true 开启调试 false 关闭调试
     - @author widuu 
     */

    public function __construct( $debug = false ){
        /* 是否开启DEBUG */
        if( $debug ) $this->debug = true;
    }

    /**
     - 获取参数,解析请求参数
     - 
     - @author widuu 
     */

    public function get_request(){
        if( !empty($_POST) ){
            // 请求的服务接口
            $this->service = $_POST['service'];
            // 获取请求字符集
            $this->charset = $_POST['charset'];
            // 获取请求的biz_content
            $request_biz_cOntent= $_POST['biz_content'];
            // 加密算法
            $this->sign_type = $_POST['sign_type'];
            // 加密字符串
            $this->sign = $_POST['sign'];
            // 如果请求格式不是Utf-8 转换格式为Utf-8
            if( strtolower($this->charset) != 'utf-8' ){
                $request_biz_cOntent= iconv('GBK', 'utf-8', $request_biz_content);
            }
            // 解析字符串为xml
            $request_xml   = @simplexml_load_string($request_biz_content, "SimpleXMLElement" , LIBXML_NOCDATA );
            // 解析为数组
            $request_array = json_decode(json_encode($request_xml),true);

            $this->request = $request_array;

            /* 解析 */
            $this->analysis($request_array);

            if($this->debug) $this->write_log('REQUEST_INFO',var_export($request_array,true));

            // 默认验证方法
            if( $this->service == 'alipay.service.check'){
                $this->verify($_POST);
                exit();
            }

            /* 返回结果 */
            return $request_array;
        }

    }

    /**
     - 回复文本内容
     - @params string content  文本数据
     - @params bool   mass     ture为群发
     - @author widuu 
     */

    public function text($content,$mass=false){
        $info['text'] = array( 'content' => $content );
        /* 组织内容 */
        $biz_cOntent= $this->common_response('text',$info,$mass);
        /* 判断是否为群发 */
        if($mass){
            $method = 'message_total';
        }else{
            $method = 'message_custom';
        }
        $sys_params = $this->common_system($method,$biz_content);
        $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params));
        /* 返回结果 结果是JSON数据 */
        $result = $this->response_post($sys_params);

        return $result;
    }

    /**
     - 回复图文内容
     - @params array articles  拼接的图文消息数组
     - @params bool   mass     ture为群发
     - @author widuu 
     */

    public function articles($articles,$mass=false){

        $info['articles'] = array($articles);
        /* 组织内容 */
        $biz_cOntent= $this->common_response('image-text',$info,$mass);
        /* 判断是否群发 */
        if($mass){
            $method = 'message_total';
        }else{
            $method = 'message_custom';
        }

        /* 加密参数 */
        $sys_params = $this->common_system($method,$biz_content);

        /* 加密字符 */
        $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params));

        /* 返回结果 结果是JSON数据 */
        $result = $this->response_post($sys_params);

        return $result;
    }

    /**
     - 关注事件
     - 
     - @author widuu 
     */

    public function is_follow(){
        $request = $this->request;
        if( $request['MsgType'] == 'event' && $request['EventType'] == 'follow' ){
            return true;
        }else{
            return false;
        }
    }

    /**
     - 取消关注事件
     - 
     - @author widuu 
     */

    public function is_unfollow(){
        $request = $this->request;
        if( $request['MsgType'] == 'event' && $request['EventType'] == 'unfollow' ){
            return true;
        }else{
            return false;
        }
    }

    /**
     - 下载用户发来的图片
     - @param  media_id string  图片id
     - @param  filename string  保存图片地址和名称
     - @author widuu 
     */

    public function down_media($media_id,$filename){
        $sys_params = $this->common_system('down_media',array('mediaId'=>$media_id));
        $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params));
        /* 返回数据 */
        $result = $this->response_post($sys_params,true);
        $result = file_put_contents($filename, $result);
        if( $this->debug ){
            $this->write_log('SAVE_IMAGE','保存图片'.(string)$result);
        }
        return $result;
    }

    /**
     - (添加|更新|获取)自定义菜单
     - @param  string $type  (add|update|get)
     - @param  array  $menu   菜单数组,如果是获取菜单可以留空
     - @author widuu 
     */

    public function menu( $type,$menu = array() ){
        if( !in_array( $type, array('get','update','add')) ){
            if( $this->debug ){
                $this->write_log('ERROR','菜单操作方法错误');
            }
            return false;
        }

        /* 拼接接口方法 */
        $method = 'menu_'.$type;
        $sys_params = $this->common_system($method,$menu);

        /* 加密字符串 */
        $sys_params['sign'] = $this->rsa_sign($this->build_query($sys_params));

        /* 请求获取结果 */
        $result = $this->response_post($sys_params);

        /* 转义并解析JSON 数据 */
        $menu_json = json_decode(iconv('GBK', 'utf-8', $result),true);

        /* 组织接口信息 */
        $interface = 'alipay_mobile_public_'.$method.'_response';

        /* 遇到错误返回 */
        if( $menu_json[$interface]['code'] != 200 ){
            if( $this->debug ){
                $this->write_log('GET_MENU_ERROR',$menu_json[$interface]['msg']);
            }
            return false;
        }

        /* 根据类型不同返回不同的结果 */
        if( $type == 'get' ){
            return $menu_json[$interface]['menu_content'];
        }else{
            return $menu_json[$interface]['msg'];
        }

    }


    /**
     - POST数据方法
     - @param  array params 参数数组
     - @author widuu 
     */

    private function response_post($params,$type=false){
        // 下载媒体和请求网关
        if($down){
            $url = 'https://openfile.alipay.com/chat/multimedia.do';
        }else{
            $url = 'https://openapi.alipay.com/gateway.do';
        }
        $ch = curl_init();
        curl_setopt($ch, CURLOPT_URL, $url);
        curl_setopt($ch, CURLOPT_HEADER, 0);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_POST, 1);
        curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($params));
        curl_setopt($ch, CURLOPT_FOLLOWLOCATION,true);
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
        curl_setopt($ch, CURLOPT_SSLVERSION, CURL_SSLVERSION_TLSv1);
        $curl = curl_exec($ch);
        curl_close($ch);
        return $curl;
    }

    /**
     - 拼接回复数据
     - @param   string $type  回复类型
     - @param   array  $info  回复内容
     - @param   bool   $mass  是否为群发
     - @author widuu 
     */

    private function common_response($type,$info,$mass=false){
        $request = $this->request;
        $params = array();
        // 如果不是群发
        if( !$mass ) $params['toUserId'] = $request['FromUserId'];
        $params['msgType'] = $type;
        $params['createTime'] = time();
        $cOntent= array_merge($params,$info);
        return $content;
    }

    /**
     - 拼接加密参数
     - @param   string $interface_type  接口类型
     - @param   array  $biz_content     返回biz_content的数组
     - @author widuu 
     */


    private function common_system($interface_type,$biz_content){

        /* 接口集合 */
        $type = $this->interface_type;

        $method = $type[$interface_type];
        /* 公共参数 */
        $params = array (
            'method' => $method,
            'charset' => 'UTF-8',
            'sign_type' => 'RSA',
            'app_id' => $this->app_id,
            'timestamp' => date ( 'Y-m-d H:i:s', time () ),
            'version'=>'1.0',
        );

        /* 获取某些接口时没有biz_content参数 */
        if( count($biz_content) > 0 ){
            $params['biz_content'] = json_encode($biz_content);
        }

        /* 返回系统参数 */
        return $params;
    }

    /**
     - 服务验证
     - @params array params  是自动获的验证信息
     - @author widuu 
     */

    private function verify($params){
        /* 参数为空 */
        if( empty($params) ){
            if( $this->debug ){
                $this->write_log('ERROR','验证参数为空');
            }
        }

        /* 构建参数,使用字典排序再拼接字符串 */
        $query_data = $this->build_query($params);

        /* 验证信息,有可能php版本BUG不支持验证 */
        $verify_result = $this->ras_verify($query_data);

        /* 返回验证结果 */
        if( $verify_result ){
            /* 取公有密钥的字符串合并为一行 */
            $public_rsa_string = file_get_contents($this->public_rsa_key_path);
            $public_rsa_string = str_replace ( "-----BEGIN PUBLIC KEY-----", "", $public_rsa_string );
            $public_rsa_string = str_replace ( "-----END PUBLIC KEY-----", "", $public_rsa_string );
            $public_rsa_string = str_replace ( "\r", "", $public_rsa_string );
            $public_rsa_string = str_replace ( "\n", "", $public_rsa_string );
            /* 构建加密字符串 */
            $response_xml = "true$public_rsa_string";
            /* 生成验证信息 */
            $sign = $this->rsa_sign (  $response_xml );
            /* 构建返回数据 */
            $respOnse= "$response_xml$signRSA";
            if( $this->debug ){
                $this->write_log('CHECK_RESPONSE',$response);
            }
            /* 输出返回信息 */
            echo $response;
            exit();
        }else{
            if( $this->debug ){
                $this->write_log('ERROR','验证失败');
            }
        }
    }

    /**
     - 拼接为字符串函数
     - @params array  params  拼接函数
     - @author widuu 
     */

    private function build_query($params){
        /* 删除sign字符串 */
        unset($params['sign']);
        /* 字典排序 */
        ksort($params);
        /* 拼接 */ 
        $query_array = array();
        foreach ($params as $k => $v) {
            $query_array[] = "$k"."="."$v";
        }
        $query_data = implode("&", $query_array);
        /* 返回拼接好的字符串 */
        return $query_data;
    }

    /**
     - 验证加密sign,有些PHP版本不支持,不支持情况直接返回true
     - @params string query_data  加密字符串
     - @author widuu 
     */

    private function ras_verify($query_data){
        /* 读取公钥文件,PEM格式 */
        $pubKey = file_get_contents($this->public_rsa_key_path);

        /* 转换为openssl格式密钥 */
        $res = openssl_get_publickey($pubKey);

        /* 调用openssl内置方法验签 */
        $result = (bool) openssl_verify($query_data, base64_decode($this->sign), $res);

        /* 释放资源 */
        openssl_free_key($res);

        /* 有些PHP版本错误,直接返回true */
        if( strpos( openssl_error_string(),'PEM_read_bio' ) ){  
             return true;
        }
        /* 返回验签结果 */
        return $result;
    }

    /**
     - 通过私有密钥加密数据
     - @params string data  加密数据
     - @author widuu 
     */

    private function rsa_sign($data) {
        /* 读取私钥 */
        $priKey = file_get_contents ( $this->private_rsa_key_path );

        /* 转换为openssl格式密钥 */
        $res = openssl_get_privatekey ( $priKey );

        /* 调用openssl 加密 */
        openssl_sign ( $data, $sign, $res );

        /* 释放资源 */
        openssl_free_key ( $res );

        /* Base64加密 */
        $sign = base64_encode ( $sign );

        /* 返回加密参数 */
        return $sign;
    }

    private function analysis($params){
        switch($params['MsgType']){
            case 'image':
                $this->media_id = $params['Image']['MediaId'];
                $this->format   = $params['Image']['Format']; 
                break;
            case 'text':
                $this->text = $params['Text']['Content'];
                break;
            case 'event':
                $this->event_type   = $params['EventType'];
                $this->action_param = $params['ActionParam'];
                break;
            default:
                break;
        }

        $this->msg_type  = $params['MsgType'];
        $this->user_info = json_decode($params['UserInfo'],true);
    }

    /**
     - DEBUG 为true时的拼接字符串
     - @param   string  $level    自定义标识符
     - @param   string  $info     自定义内容
     - @param   string  $log_path 自定义日志路径
     - @author widuu 
     */

    public function write_log($level,$info,$log_path = '' ){
        if( empty($log_path) ){
            $log_path = dirname ( __FILE__ ) . "/log.txt";
        }
        file_put_contents($log_path, "[$level]".date ( "Y-m-d H:i:s" ) . "  " . $info . "\r\n", FILE_APPEND );
    }
}


推荐阅读
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • VScode格式化文档换行或不换行的设置方法
    本文介绍了在VScode中设置格式化文档换行或不换行的方法,包括使用插件和修改settings.json文件的内容。详细步骤为:找到settings.json文件,将其中的代码替换为指定的代码。 ... [详细]
  • 本文介绍了Java工具类库Hutool,该工具包封装了对文件、流、加密解密、转码、正则、线程、XML等JDK方法的封装,并提供了各种Util工具类。同时,还介绍了Hutool的组件,包括动态代理、布隆过滤、缓存、定时任务等功能。该工具包可以简化Java代码,提高开发效率。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • t-io 2.0.0发布-法网天眼第一版的回顾和更新说明
    本文回顾了t-io 1.x版本的工程结构和性能数据,并介绍了t-io在码云上的成绩和用户反馈。同时,还提到了@openSeLi同学发布的t-io 30W长连接并发压力测试报告。最后,详细介绍了t-io 2.0.0版本的更新内容,包括更简洁的使用方式和内置的httpsession功能。 ... [详细]
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • 使用在线工具jsonschema2pojo根据json生成java对象
    本文介绍了使用在线工具jsonschema2pojo根据json生成java对象的方法。通过该工具,用户只需将json字符串复制到输入框中,即可自动将其转换成java对象。该工具还能解析列表式的json数据,并将嵌套在内层的对象也解析出来。本文以请求github的api为例,展示了使用该工具的步骤和效果。 ... [详细]
  • 本文介绍了高校天文共享平台的开发过程中的思考和规划。该平台旨在为高校学生提供天象预报、科普知识、观测活动、图片分享等功能。文章分析了项目的技术栈选择、网站前端布局、业务流程、数据库结构等方面,并总结了项目存在的问题,如前后端未分离、代码混乱等。作者表示希望通过记录和规划,能够理清思路,进一步完善该平台。 ... [详细]
  • Android JSON基础,音视频开发进阶指南目录
    Array里面的对象数据是有序的,json字符串最外层是方括号的,方括号:[]解析jsonArray代码try{json字符串最外层是 ... [详细]
  • jmeter实践:从csv中获取带引号的数据详情的技巧和运行全部数据的方法
    本文分享了jmeter实践中从csv中获取带引号的数据的解决办法,包括设置CSV Data Set Config和运行脚本获取数据的方法。另外还介绍了循环运行csv中全部数据的解决方法,避免每次修改csv用例都需要修改脚本的麻烦。通过了解和掌握工具的细节点,可以更好地解决问题和提高技术水平。 ... [详细]
  • 单点登录原理及实现方案详解
    本文详细介绍了单点登录的原理及实现方案,其中包括共享Session的方式,以及基于Redis的Session共享方案。同时,还分享了作者在应用环境中所遇到的问题和经验,希望对读者有所帮助。 ... [详细]
  • 本文讨论了Kotlin中扩展函数的一些惯用用法以及其合理性。作者认为在某些情况下,定义扩展函数没有意义,但官方的编码约定支持这种方式。文章还介绍了在类之外定义扩展函数的具体用法,并讨论了避免使用扩展函数的边缘情况。作者提出了对于扩展函数的合理性的质疑,并给出了自己的反驳。最后,文章强调了在编写Kotlin代码时可以自由地使用扩展函数的重要性。 ... [详细]
  • 使用正则表达式爬取36Kr网站首页新闻的操作步骤和代码示例
    本文介绍了使用正则表达式来爬取36Kr网站首页所有新闻的操作步骤和代码示例。通过访问网站、查找关键词、编写代码等步骤,可以获取到网站首页的新闻数据。代码示例使用Python编写,并使用正则表达式来提取所需的数据。详细的操作步骤和代码示例可以参考本文内容。 ... [详细]
author-avatar
邱文馨4966
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有