【網站支付PHP篇】thinkPHP整合支付寶支付(擔保交易)
目錄
- 系列說明
- 開發環境
- 部署支付寶
- 支付請求
- 支付寶返回處理
系列說明
最近在幫朋友的系統安裝支付模組(兌換網站積分),現在總結一些開發心得,希望對大家有用。這個系列會講以下第三方支付平臺的整合:
以後有更新其他平臺也會寫出來。
這次的支付系統是用來網站的積分充值。為了簡化,這裡採用的是1元RMB=1個網站積分的兌換。
這裡,不會描述怎麼申請支付平臺的收款帳號。
開發環境
ThinkPHP 3.1.2 (我用的是標準版)
win7 + Apache2
mysql 5.1
這裡說一下資料庫的設計。
部署支付寶
我用的是支付寶擔保交易(即時到賬的門檻有點高=.=).先從支付寶網站下載擔保交易的demo(要注意編碼,我用的是utf-8)。
前往ThinkPHP目錄下的Extend,找到Vendor目錄(這個目錄是第三方類庫的存放目錄,如果沒有,手動建立一個,要標準版本里面是沒有的),然後在Vendor下建立Alipay目錄存在支付寶相關的檔案。目錄結構如下:
接著修改 alipay.config.php 檔案:
支付請求
配置好支付寶後,就可以進行支付請求了。支付寶的示例程式碼很是詳細,很多可以直接用。
首先,建立OrderAction.class.php 來處理訂單相關的業務操作。
定義pay方法:
/** * 支付訂單 */ public function pay(){ $this->checkLogin(); header('Content-type: text/html; charset=utf-8'); $id = args("id", 0); $DAO = new OrderModel(); $order = $DAO->where("id=".$id)->find(); $error = ""; if(!isset($order)){ $error = "訂單不存在"; }else if($order['status'] == 1){ $error = "此訂單已經完成,無需再次支付!"; } else if($order['status'] == 2){ $error = "此訂單已經取消,無法支付,請重新下單!"; } if($error != ""){ $this->_FAIL("系統錯誤",$error,$this->getErrorLinks()); return ; } //支付寶 if($order['payment'] == 'alipay'){ $this->payWithAlipay($order); } else if($order['payment'] == 'ecpss'){ $this->payWithEcpss($order); } else if($order['payment'] == 'dinpay'){ $this->payWithDinpay($order); } }
然後再定義 payWithAlipay($order) 方法(根據具體情況修改相關的引數即可):
/** * 以支付寶形式支付 * @param unknown_type $order */ private function payWithAlipay($order){ //引入支付寶相關的檔案 require_once(VENDOR_PATH."Alipay/alipay.config.php"); require_once(VENDOR_PATH."Alipay/lib/alipay_submit.class.php"); //支付型別 $payment_type = "1"; //必填,不能修改 //伺服器非同步通知頁面路徑 $notify_url = C("HOST")."index.php/Order/notifyOnAlipay"; //頁面跳轉同步通知頁面路徑 $return_url = C("HOST")."index.php/Order"; //賣家支付寶帳戶 $seller_email = $alipay_config['seller_email']; //必填 //商戶訂單號, 從訂單物件中獲取 $out_trade_no = $order['tradeNo']; //商戶網站訂單系統中唯一訂單號,必填 //訂單名稱 $subject = $order['subject']; //必填 //付款金額 $price = $order['price']; //必填 $body = $order['subject']; //商品展示地址 $show_url = C('HOST'); //構造要請求的引數陣列,無需改動 $parameter = array( "service" => "create_partner_trade_by_buyer", "partner" => trim($alipay_config['partner']), "payment_type"=> $payment_type, "notify_url"=> $notify_url, "return_url"=> $return_url, "seller_email"=> $seller_email, "out_trade_no"=> $out_trade_no, "subject"=> $subject, "price"=> $price, "quantity"=> "1", "logistics_fee"=> "0.00", "logistics_type"=> "EXPRESS", "logistics_payment"=> "SELLER_PAY", "body"=> $body, "show_url"=> $show_url, "receive_name"=> "", "receive_address"=> "", "receive_zip"=> "", "receive_phone"=> "", "receive_mobile"=> "", "_input_charset"=> trim(strtolower($alipay_config['input_charset'])) ); //建立請求 $alipaySubmit = new AlipaySubmit($alipay_config); $html_text = $alipaySubmit->buildRequestForm($parameter,"get", "去支付"); echo $html_text; }
支付寶返回處理
最後是支付寶非同步通知的處理函式(這裡,我只對非同步返回作處理):
/**
* 支付寶非同步通知
*/
public function notifyOnAlipay(){
require_once(VENDOR_PATH."Alipay/alipay.config.php");
require_once(VENDOR_PATH."Alipay/lib/alipay_notify.class.php");
$orderLogDao = new OrderLogModel();
//計算得出通知驗證結果
$alipayNotify = new AlipayNotify($alipay_config);
$verify_result = $alipayNotify->verifyNotify();
if($verify_result) {//驗證成功
//商戶訂單號
$out_trade_no = $_POST['out_trade_no'];
//支付寶交易號
$trade_no = $_POST['trade_no'];
//根據訂單號獲取訂單
$DAO = new OrderModel();
$order = $DAO->where("tradeNo='".$out_trade_no."'")->find();
//如果訂單不存在,設定為0
if(!isset($order)){
$orderId = 0;
}
else{
$orderId = $order['id'];
}
//交易狀態
$trade_status = $_POST['trade_status'];
$log = "notify from Alipay, trade_status=".$trade_status." alipay sign=".$_POST['sign'];
$orderLog['order_id'] = $orderId;
$orderLog['addDate'] = sqlDate();
if($_POST['trade_status'] == 'WAIT_BUYER_PAY') {
//該判斷表示買家已在支付寶交易管理中產生了交易記錄,但沒有付款
}
/*
* 成功付款後,進行積分操作
*/
else if($_POST['trade_status'] == 'WAIT_SELLER_SEND_GOODS') {
//該判斷表示買家已在支付寶交易管理中產生了交易記錄且付款成功,但賣家沒有發貨
if(isset($order) && $order['status'] == 0){
$resultInfo = $this->doAfterPaySuccess($DAO, $order);
$log.= $resultInfo;
}
}
else if($_POST['trade_status'] == 'WAIT_BUYER_CONFIRM_GOODS') {
//該判斷表示賣家已經發了貨,但買家還沒有做確認收貨的操作
}
else if($_POST['trade_status'] == 'TRADE_FINISHED') {
//該判斷表示買家已經確認收貨,這筆交易完成
}
else {
}
/*
* 儲存orderlog
*/
$orderLog['log'] = $log;
$orderLogDao->add($orderLog);
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
echo "success"; //返回成功標記給支付寶
}
else {
//驗證不通過時,也記錄下來
$orderLog['log'] = "notify from Alipay, 但是驗證不通過,sign=".$_POST['sign'];
$orderLog['order_id'] = -1;
$orderLog['addDate'] = sqlDate();
$orderLogDao->add($orderLog);
//驗證失敗
echo "fail";
}
}
測試一筆,得到如下的結果:
總結
支付寶的整合還是比較簡單的,官方的教程很詳細。
這裡要注意的是,
1、alipay.config.php 下的 cacert.pem 檔案的路徑一定要對,之前我沒注意這點,結果出來以下的錯誤:
2、在notify_url中,一定只能返回 "success"或者"fail",也不能對這個url作登入驗證的處理。
3、還要注意對網站訂單的反重複處理
如有需要整合服務的朋友可以到小弟的店裡看看:ThinkPHP支付整合 (溫馨提示:此連結為淘寶連結)