欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  php教程

ThinkPHP整合支付宝担保交易

程序员文章站 2024-02-14 21:37:40
...
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 控制器,然后处理代码: 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");本地测试异步中写操作数据库是没任何意义的

因为异步需要服务器上测试才行的。
打个广告
个人博客 青春博客 www.loveteemo.com 欢迎大家来访
原本链接:http://loveteemo.com/article-130.html

AD:真正免费,域名+虚机+企业邮箱=0元