ThinkPHP整合支付宝担保交易

jerry thinkphp 2015年11月21日 收藏
ThinkPHP整合支付宝担保交易
本代码参考大神 http://www.thinkphp.cn/code/240.html 的思路

1.登陆支付宝后台,下载担保交易的集成包。


2.下载完成后的文件说明:

纯担保交易接口-create_partner_trade_by_buyer(20151015)
确认发货接口-send_goods_confirm_by_platform(20150312)
根据自己需要去选择,需要说明下,先担保整合完成后才回去处理确认发货,因为确认发货时需要担保交易的支付宝交易编号

对应的代码文件结构
───────
create_partner_trade_by_buyer-php-UTF-8
  │
  ├lib┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈类文件夹
  │  │
  │  ├alipay_core.function.php ┈┈┈┈┈┈支付宝接口公用函数文件
  │  │
  │  ├alipay_notify.class.php┈┈┈┈┈┈┈支付宝通知处理类文件
  │  │
  │  ├alipay_submit.class.php┈┈┈┈┈┈┈支付宝各接口请求提交类文件
  │  │
  │  └alipay_md5.function.php┈┈┈┈┈┈┈支付宝接口MD5函数文件
  │
  ├log.txt┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈日志文件
  │
  ├alipay.config.php┈┈┈┈┈┈┈┈┈┈┈┈基础配置类文件
  │
  ├alipayapi.php┈┈┈┈┈┈┈┈┈┈┈┈┈┈支付宝接口入口文件
  │
  ├notify_url.php ┈┈┈┈┈┈┈┈┈┈┈┈┈服务器异步通知页面文件
  │
  ├return_url.php ┈┈┈┈┈┈┈┈┈┈┈┈┈页面跳转同步通知文件
  │
  ├cacert.pem ┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈用于CURL中校验SSL的CA证书文件
  │
  └readme.txt ┈┈┈┈┈┈┈┈┈┈┈┈┈┈┈使用说明文本
这里我们先处理代码

3.核心处理代码

把lib目录文件下的4个核心文件放入 ThinkPHP/Library/Vendor/Alipay 下

修改文件名为:
alipay_core.function.php       ->        Corefunction.php
alipay_notify.class.php        ->        Notify.php
alipay_submit.class.php        ->        Submit.php
alipay_md5.function.php        ->        Md5function.php
其中 Notify.php 和 Md5function.php 需要删除前面引入的两行代码
require_once("alipay_core.function.php");
require_once("alipay_md5.function.php");
因为在使用TP第三方扩展类的时候会自动引入他需要的这两个文件

核心框架整合完成之后我们来整理逻辑代码。

4.逻辑代码整理

首先我们把 签名文件 cacert.pem 放在网站的跟目录,其他目录也行,不过需要有访问权限的

然后在公共配置文件conf.php中添加 支付宝配置
    //支付宝配置参数
'alipay_config'=>array(
    'partner' =>'2088**********************',   //这里是你在成功申请支付宝接口后获取到的PID;
    'key'=>'ob4x7k0*************************',//这里是你在成功申请支付宝接口后获取到的Key
    'sign_type'=>strtoupper('MD5'),
    'input_charset'=> strtolower('utf-8'),
    'cacert'=> getcwd().'\\cacert.pem',//liunx这里需要注意 \\ 和 / 在liunx的区别
    'transport'=> 'http',
    'seller_email'=>'775919499@qq.com',// 这里是你的收款账号,
),
//以上配置项,是从接口包中alipay.config.php 文件中复制过来,进行配置;
    'alipay'   =>array(
//这里是异步通知页面url,提交到项目的Pay控制器的notifyurl方法;
'notify_url'=>'http://www.loveteemo.com/Pay/notifyurl', 
//这里是页面跳转通知url,提交到项目的Pay控制器的returnurl方法;
'return_url'=>'http://www.loveteemo.com/Pay/returnurl',
),
然后去创建一个 Key 控制器,然后处理代码:
<?php
namespace Home\Controller;
use Think\Controller;
class KeyController extends Controller {
    //利用构造函数引入核心文件
    public function _initialize() {
        vendor('Alipay.Corefunction');
        vendor('Alipay.Md5function');
        vendor('Alipay.Notify');
        vendor('Alipay.Submit');    
    }
  //首页,用来展示商品页
    public function index(){
        $this->display();
    }
    //订单页,我写死了的,可以根据自己需要进行修改
    public function order(){
        $type = I('get.type');
        if($type==1){
        $date = array('type'=>1,"price"=>'0.01',"name"=>"《火星救援》");
        }elseif($type==2){
        $date = array('type'=>2,"price"=>'0.01',"name"=>"《死神的精确度》");
        }elseif($type==3){
        $date = array('type'=>3,"price"=>'0.01',"name"=>"《寂寞是毒,也是解药》");
        }elseif($type==4){
        $date = array('type'=>4,"price"=>'0.01',"name"=>"《只要不忘了回家的路》");
        }elseif($type==5){
        $date = array('type'=>5,"price"=>'0.01',"name"=>"《异想星球 hello,我是托比小黑》");
        }elseif($type==6){
        $date = array('type'=>6,"price"=>'0.01',"name"=>"《张鸣说历史:大国的虚与实》");
        }
        $this->time = time();
        $this->assign("data",$date);
        $this->display();
    }
    //订单页点击提交,传递必要参数后开始支付,可自行修改
    public function payorder(){
        //传递数组配置
        $alipay_config=C('alipay_config');
        /**************************请求参数**************************/
            //支付类型
        $payment_type = "1";        //必填,不能修改
        //服务器异步通知页面路径
        $notify_url = C('alipay.notify_url');        //需http://格式的完整路径,不能加?id=123这类自定义参数
        //页面跳转同步通知页面路径
        $return_url = C('alipay.return_url');         //需http://格式的完整路径,不能加?id=123这类自定义参数,不能写成http://localhost/
        //商户订单号
        $out_trade_no = $_POST['orderid'];        //商户网站订单系统中唯一订单号,必填
        //订单名称
        $subject = $_POST['ordername'];        //必填
        //付款金额
        $price = $_POST['orderprice'];        //必填
        //商品数量
        $quantity = "1";        //必填,建议默认为1,不改变值,把一次交易看成是一次下订单而非购买一件商品
        //物流费用
        $logistics_fee = "0.00";        //必填,即运费
        //物流类型
        $logistics_type = "EXPRESS";        //必填,三个值可选:EXPRESS(快递)、POST(平邮)、EMS(EMS)
        //物流支付方式
        $logistics_payment = "SELLER_PAY";        //必填,两个值可选:SELLER_PAY(卖家承担运费)、BUYER_PAY(买家承担运费)
        //订单描述
        $body = $_POST['orderbody'];
            //商品展示地址
        $show_url = $_POST['ordershow'];        //需以http://开头的完整路径,如:http://www.商户网站.com/myorder.html
        //收货人姓名
        $receive_name = $_POST['user_name'];        //如:张三
        //收货人地址
        $receive_address = $_POST['user_address'];        //如:XX省XXX市XXX区XXX路XXX小区XXX栋XXX单元XXX号
        //收货人邮编
        $receive_zip = $_POST['user_zip'];        //如:123456
        //收货人电话号码
        $receive_phone = $_POST['user_phone'];        //如:0571-88158090
        //收货人手机号码
        $receive_mobile = $_POST['user_mobile'];        //如:13312341234
        /************************************************************/
    
        //构造要请求的参数数组,无需改动
            $parameter = array(
            "service" => "create_partner_trade_by_buyer",
            "partner" => trim($alipay_config['partner']),
            "seller_email" => trim($alipay_config['seller_email']),
            "payment_type"=> $payment_type,
            "notify_url"=> $notify_url,
            "return_url"=> $return_url,
            "out_trade_no"=> $out_trade_no,
            "subject"=> $subject,
            "price"=> $price,
            "quantity"=> $quantity,
            "logistics_fee"=> $logistics_fee,
            "logistics_type"=> $logistics_type,
            "logistics_payment"=> $logistics_payment,
            "body"=> $body,
            "show_url"=> $show_url,
            "receive_name"=> $receive_name,
            "receive_address"=> $receive_address,
            "receive_zip"=> $receive_zip,
            "receive_phone"=> $receive_phone,
            "receive_mobile"=> $receive_mobile,
            "_input_charset"=> trim(strtolower($alipay_config['input_charset']))
            );
            //存入数据库订单信息 static为99是无效订单
            M('test')->add(array("orderid"=>$out_trade_no,"addtime"=>time(),"ordername"=>$subject,"orderprice"=>$price,"static"=>99));
            //建立请求
            $alipaySubmit = new \AlipaySubmit($alipay_config);
            //dump($alipaySubmit);die;
            $html_text = $alipaySubmit->buildRequestForm($parameter,"get", "确认");
            echo $html_text;
            }
    /******************************    服务器异步通知页面方法    *******************************/
    public function notifyurl(){
        //防止乱码
        header("Content-Type:text/html;charset=utf-8");
        //计算得出通知验证结果
        $alipay_config=C('alipay_config');
        $alipayNotify = new \AlipayNotify($alipay_config);
        $verify_result = $alipayNotify->verifyNotify();
        if($verify_result) {//验证成功
        //商户订单号
        logResult("订单编号:".$_POST['out_trade_no'].",状态".$_POST['trade_status']."");
        $out_trade_no = $_POST['out_trade_no'];
        //支付宝交易号
        $trade_no = $_POST['trade_no'];
        //交易状态
        $trade_status = $_POST['trade_status'];
        if($_POST['trade_status'] == 'WAIT_BUYER_PAY') {
        //该判断表示买家已在支付宝交易管理中产生了交易记录,但没有付款
        //判断该笔订单是否在商户网站中已经做过处理
        //如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
        //如果有做过处理,不执行商户的业务程序
                echo "success";//请不要修改或删除
                //调试用,写文本函数记录程序运行情况是否正常
                //获取支付宝的订单号后写入数据库,修改订单状态为0 待支付
               M('test')->where(array("orderid"=>$out_trade_no))->save(array("static"=>0,"lasttime"=>time(),'payid'=>$trade_no));
               //文本日志文件,这里的日志文件会在网站根目录生成一个log.txt文件
                logResult("这里是等待付款");
            }
        else if($_POST['trade_status'] == 'WAIT_SELLER_SEND_GOODS') {
        //该判断表示买家已在支付宝交易管理中产生了交易记录且付款成功,但卖家没有发货
        //判断该笔订单是否在商户网站中已经做过处理
        //买家支付后,修改状态为已支付代发货
        M('test')->where(array("orderid"=>$out_trade_no))->save(array("static"=>1,"lasttime"=>time()));
        //如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
        //如果有做过处理,不执行商户的业务程序
                echo "success";//请不要修改或删除
                //调试用,写文本函数记录程序运行情况是否正常
                logResult("支付完成!订单编号:".$out_trade_no.",状态".$_POST['trade_status']."");
            }
        else if($_POST['trade_status'] == 'WAIT_BUYER_CONFIRM_GOODS') {
        //该判断表示卖家已经发了货,但买家还没有做确认收货的操作
        //判断该笔订单是否在商户网站中已经做过处理
        //发货完成后会修改状态
        M('test')->where(array("orderid"=>$out_trade_no))->save(array("static"=>2,"lasttime"=>time()));
        //如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
        //如果有做过处理,不执行商户的业务程序
                echo "success";//请不要修改或删除
                //调试用,写文本函数记录程序运行情况是否正常
                logResult("发货完成,等待买家收货");
            }
        else if($_POST['trade_status'] == 'TRADE_FINISHED') {
        //该判断表示买家已经确认收货,这笔交易完成
        //判断该笔订单是否在商户网站中已经做过处理
        //买家收货后订单完成
        M('test')->where(array("orderid"=>$out_trade_no))->save(array("static"=>3,"lasttime"=>time()));
        //如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
        //如果有做过处理,不执行商户的业务程序
                echo "success";//请不要修改或删除
                //调试用,写文本函数记录程序运行情况是否正常
                logResult("交易完成!");
            }
            else {
        //其他状态判断
                echo "success";
                //调试用,写文本函数记录程序运行情况是否正常
                logResult ("错误");
            }
        }
        else {
            //验证失败
            echo "fail";
            //调试用,写文本函数记录程序运行情况是否正常
            logResult("验证失败");
        }
    }
    
    /*        页面跳转处理方法;        */
    public function returnurl(){
        $alipay_config=C('alipay_config');
         //计算得出通知验证结果
        $alipayNotify = new \AlipayNotify($alipay_config);
        $verify_result = $alipayNotify->verifyReturn();
        if($verify_result) {//验证成功
        //——请根据您的业务逻辑来编写程序(以下代码仅作参考)——
            //获取支付宝的通知返回参数,可参考技术文档中页面跳转同步通知参数列表
        //商户订单号
        $out_trade_no = $_GET['out_trade_no'];
        //支付宝交易号
        $trade_no = $_GET['trade_no'];
        //交易状态
        $trade_status = $_GET['trade_status'];
            if($_GET['trade_status'] == 'WAIT_SELLER_SEND_GOODS') {
        //判断该笔订单是否在商户网站中已经做过处理
        //如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
        //如果有做过处理,不执行商户的业务程序
            }
        //
            else {
              echo "trade_status=".$_GET['trade_status'];
            }
        $this->assign("payid",$trade_no);
        }
        else {
            //验证失败
            //如要调试,请看alipay_notify.php页面的verifyReturn函数
            echo "验证失败";
        }
        $this->display();
}
//自动发货 利用隐藏表单传递必须数据过来
       public function sendgoods(){
            $alipay_config=C('alipay_config');
             /**************************请求参数**************************/
        //支付宝交易号
        $trade_no = $_POST['WIDtrade_no'];
        //必填
        //物流公司名称
        $logistics_name = $_POST['WIDlogistics_name'];
        //必填
        //物流发货单号
        $invoice_no = $_POST['WIDinvoice_no'];
        //物流运输类型
        $transport_type = $_POST['WIDtransport_type'];
        //三个值可选:POST(平邮)、EXPRESS(快递)、EMS(EMS)
        /************************************************************/
        //构造要请求的参数数组,无需改动
        $parameter = array(
        "service" => "send_goods_confirm_by_platform",
        "partner" => trim($alipay_config['partner']),
        "trade_no"=> $trade_no,
        "logistics_name"=> $logistics_name,
        "invoice_no"=> $invoice_no,
        "transport_type"=> $transport_type,
        "_input_charset"=> trim(strtolower($alipay_config['input_charset']))
        );
        //建立请求
        $alipaySubmit = new \AlipaySubmit($alipay_config);
        $html_text = $alipaySubmit->buildRequestHttp($parameter);
        //解析XML
        //注意:该功能PHP5环境及以上支持,需开通curl、SSL等PHP配置环境。建议本地调试时使用PHP开发软件
        $doc = new \DOMDocument();
        $doc->loadXML($html_text);
        //请在这里加上商户的业务逻辑程序代码
        //——请根据您的业务逻辑来编写程序(以下代码仅作参考)——
        //获取支付宝的通知返回参数,可参考技术文档中页面跳转同步通知参数列表
        //解析XML
        if( ! empty($doc->getElementsByTagName( "alipay" )->item(0)->nodeValue) ) {
        $alipay = $doc->getElementsByTagName( "alipay" )->item(0)->nodeValue;
        //echo $alipay;
        //M('test')->where(array("orderid"=>$out_trade_no))->save(array("static"=>2,"lasttime"=>time()));
        $this->success("自动发货完成!","/Home/Key/lists");
        }
}
    //筛选无效订单后展示
    public function lists(){
        $lists = M('test')->where("static != 99")->limit(20)->order("id desc")->select();
        $this->assign("lists",$lists);
        $this->display();
    }
}[code]到这里基本的业务逻辑就完成了,附带测试的数据库给大家分享下[code]CREATE TABLE `web_test` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `orderid` varchar(16) NOT NULL,
  `ordername` varchar(128) NOT NULL,
  `orderprice` varchar(16) NOT NULL,
  `static` int(11) NOT NULL COMMENT '0为未支付,1为已支付未发货,2为发货为确认收,3为确认收,4为取消',
  `addtime` int(11) NOT NULL,
  `lasttime` int(11) NOT NULL,
  `payid` varchar(32) NOT NULL,
  PRIMARY KEY (`id`),
  UNIQUE KEY `orderid` (`orderid`)
) ENGINE=InnoDB AUTO_INCREMENT=43 DEFAULT CHARSET=utf8 AUTO_INCREMENT=1 ;
5.需要注意的地方:

签名的目录,上面有说过,签名的目录在liunx和windows是有区别的,liunx需要改成
 'cacert'=> getcwd().'/cacert.pem',
不然会报错说签名没找到



在部署完成后测试的时候会遇到有些浏览器乱码,谷歌正常

这里需要注意的是TP在异步的时候会出现,在urldecode的时候中文出现乱码,所以在这里我在前面加一行代码防止乱码
header("Content-Type:text/html;charset=utf-8");
本地测试异步中写操作数据库是没任何意义的

因为异步需要服务器上测试才行的。