PHP网上支付

时间:2024-02-16 14:38:20

1,网上支付方式两类:企业与银行对接和通过中间公司间接与银行对接。

(1),企业与银行对接,优点:因为直接与银行进行财务结算,交易资金结算比较安全。适合资金流量比较大的企业,这种方案适合于,每月结算金额百万以上的企业。缺点:开发工作量比较大,而且银行不定期升级系统,随着银行系统的升级,企业也需要作相应改动,所以维护工作量比较大,而且企业每年需要向银行交纳一定数量的接口使用费。

(2),通过中间公司间接与银行对接,优点:开发工作量少,因为使用的是中间企业提供的接入规范,所以银行升级系统不需要企业作相应修改,除非中间企业的接入规范发生改变,企业才做修改,相对于前一种,这种方案的维护工作量比较少,因为只与一家企业对接,所以接入费用相对比较低,这种方案适合于:每月结算金额在几十万以下的中小企业。缺点:因为是与中间企业进行资金结算,目前所有中间企业都是私企,资金安全是个大问题。

2,易宝和支付宝不同的

买家通过易宝直接把钱转账给卖家的银行账号;

买家把钱打给支付宝,在确认卖家发货后,支付宝再把钱转到卖家的银行账号;

3,易宝支付协议介绍

(1)     支付请求时HTTPS协议请求(https协议是http协议的安全版本),商户以GET或POST方式发送到易宝支付。

(2)     易宝支付平台统一使用GBK/GB2312编码方式。

(3)    参数名称和参数说明中规定的固定值必须与列表中完全一致(大小写敏感)。

4,学习下php通过易宝支付过程,其流程如下:

(1)使用测试账号作为易宝的商家,假设测试账号是”卖家1“。最好下载一个易宝支付产品接口文档。

$p1_MerId = "10001126856";
$merchantKey = "69cl522AV6q613Ii4W6u8K6XuW8vM1N6bFgyv769220IuYe9u37N4y7rI4Pl";

(2)程序框架如下图所示:

(3)代码页内容

① pay.php购买页面,其中“pd_FrpId”表示支付通道列表,需要去易宝帮助文档的“支付通道编码列表”查看每个银行的相应代码。

<html>
<body>
    <form method="post" action="payConfirm.php">
        订单号:<input type="text" name="p2_Order" /><br/>
        支付金额:<input type="text" name="p3_Amt" /><br/>
        请选择支付银行:<br/>
        <input type="radio" name="pd_FrpId" value="CMBCHINA-NET" />招商银行<br/>
        <input type="radio" name="pd_FrpId" value="ICBC-NET" />工商银行<br/>
        <input type="radio" name="pd_FrpId" value="ABC-NET" />农业银行<br/>
        <input type="radio" name="pd_FrpId" value="CCB-NET" />建设银行<br/>
        <input type="radio" name="pd_FrpId" value="GDB-NET" />广发银行<br/>
        <input type="submit" value="确认支付" />
    </form>
</body>
</html>

② payConfirm.php,支付确认页面,此页面需要根据易宝文档中”支付请求参数“部分,将各参数赋值,同时请求页面是“https://www.yeepay.com/app-merchant-proxy/node”。

<?php
require_once "common.php";
$p0_Cmd ="Buy";
$p1_MerId = "10001126856";

$p2_Order = $_POST[\'p2_Order\'];
$p3_Amt = $_POST[\'p3_Amt\'];
$p4_Cur = "CNY";
$p5_Pid = "";  //商品名称
$p6_Pcat = ""; //商品种类
$p7_Pdesc = ""; //商品描述

$p8_Url = "http://www.lhycentos.com:2080/myNetpayment/res.php"; //支付成功页面
$p9_SAF = "0"; //送货地址
$pa_MP = ""; //商品扩展信息

$pr_NeedResponse = "1"; //商家是否返回易宝成功信息

$pd_FrpId = $_POST[\'pd_FrpId\']; //各种银行卡支付通道

//把请求参数拼接
$data = $p0_Cmd.$p1_MerId.$p2_Order.$p3_Amt.$p4_Cur.$p5_Pid.$p6_Pcat.$p7_Pdesc.$p8_Url.$p9_SAF.$pa_MP.$pd_FrpId.$pr_NeedResponse;

$merchantKey = "69cl522AV6q613Ii4W6u8K6XuW8vM1N6bFgyv769220IuYe9u37N4y7rI4Pl";
//hmac是签名单,是用于易宝和商家互相确认的关键字
//这里需要我们使用算法来生成(md5-hmac算法)
$hmac = HmacMd5($data,$merchantKey);
?>

<html>
<body>
您的订单号是<?=$p2_Order ?>,订单金额是<?=$p3_Amt ?>
<form method="post" action="https://www.yeepay.com/app-merchant-proxy/node">
    <input type="hidden" name="p0_Cmd" value="<?=$p0_Cmd?>" />
    <input type="hidden" name="p1_MerId" value="<?=$p1_MerId?>" />
    <input type="hidden" name="p2_Order" value="<?=$p2_Order?>" />
    <input type="hidden" name="p3_Amt" value="<?=$p3_Amt?>" />
    <input type="hidden" name="p4_Cur" value="<?=$p4_Cur?>" />
    <input type="hidden" name="p5_Pid" value="<?=$p5_Pid?>" />
    <input type="hidden" name="p6_Pcat" value="<?=$p6_Pcat?>" />
    <input type="hidden" name="p7_Pdesc" value="<?=$p7_Pdesc?>" />
    <input type="hidden" name="p8_Url" value="<?=$p8_Url?>" />
    <input type="hidden" name="p9_SAF" value="<?=$p9_SAF?>" />
    <input type="hidden" name="pa_MP" value="<?=$pa_MP?>" />
    <input type="hidden" name="pd_FrpId" value="<?=$pd_FrpId?>" />
    <input type="hidden" name="pr_NeedResponse" value="<?=$pr_NeedResponse?>" />
    <input type="hidden" name="hmac" value="<?=$hmac?>" />
    <input type="submit" value="确认支付" />
</form>
</body>
</html>

③ payConfirm.php页面中使用到编码函数,此函数放在common.php中,common.php内容如下:

<?php

function HmacMd5($data,$key){

    $key = iconv("GB2312","UTF-8",$key);
    $data = iconv("GB2312","UTF-8",$data);
    $b = 64;
    if(strlen($key) > $b){
        $key = pack("H*",md5($key));
    }
    $key = str_pad($key,$b,chr(0x00));
    $ipad = str_pad(\'\',$b,chr(0x36));
    $opad = str_pad(\'\',$b,chr(0x5c));
    $k_ipad = $key ^ $ipad;
    $k_opad = $key ^ $opad;
    return md5($k_opad . pack("H*",md5($k_ipad . $data)));
}
?>

④ payConfirm.php将支付数据发送到“https://www.yeepay.com/app-merchant-proxy/node”后,进入支付页面,易宝从用户的网银中把钱转到“卖家1”的账户中,如果操作成功,返回到res.php页面,此页面是在payConfirm.php中$p8_Url = "http://www.lhycentos.com:2080/myNetpayment/res.php"; 指定的。

<?php
echo "支付成功!":
?>

(4)现在实现了支付,但是存在一个漏洞,如果有黑客攻击,修改了支付页面,会将钱打到错误的人账号中。在res.php页面中根据收到的参数,重新编码和支付页面发送过来的参数进行比较,如果相同表示支付成功,如果不相同,表示支付失败。res.php的代码可以修改成如下方式:

<?php
require_once "common.php";
$p1_MerId = "10001126856";

$r0_Cmd = $_REQUEST[\'r0_Cmd\'];
$r1_Code = $_REQUEST[\'r1_Code\'];
$r2_TrxId = $_REQUEST[\'r2_TrxId\'];
$r3_Amt = $_REQUEST[\'r3_Amt\'];
$r4_Cur = $_REQUEST[\'r4_Cur\'];
$r5_Pid = $_REQUEST[\'r5_Pid\'];
$r6_Order = $_REQUEST[\'r6_Order\'];
$r7_Uid = $_REQUEST[\'r7_Uid\'];
$r8_MP = $_REQUEST[\'r8_MP\'];
$r9_BType = $_REQUEST[\'r9_BType\'];
$hmac = $_REQUEST[\'hmac\'];


//把请求参数拼接
$res_src = $p1_MerId.$r0_Cmd.$r1_Code.$r2_TrxId.$r3_Amt.$r4_Cur.$r5_Pid.$r6_Order.$r7_Uid.$r8_MP.$r9_BType;
$merchantKey = "69cl522AV6q613Ii4W6u8K6XuW8vM1N6bFgyv769220IuYe9u37N4y7rI4Pl";

//对返回的结果进行md5-hmac加密处理,和返回的hmac签名串比较
if(HmacMd5($res_src,$merchantKey) == $hmac){
    if($r1_Code == 1){  //支付结果,1代表成功
        if($r9_BType == 1){
            echo "交易成功!";
            echo "订单号为".$r6_Order."支付成功!所付金额是".$r3_Amt."易宝支付订单号是".$r2_TrxId."。";
            echo "<br/>浏览器重定向";
        }elseif($r9_BType == 1){
            echo "success";
            echo "<br/>交易成功!";
            echo "<br/>服务器点对点通讯";
        }
    }
}else{
    echo "签名被篡改";
}
?>

4,支付过程存在的问题, 有个页面执行sql语句,如果sql语句这样:update 数据表 set 字段+100,必须对sql的执行加限制条件。因为反复刷新页面,可以导致sql语句多次执行。比如:

if(数据表1的status = 0){

update 数据表1 set 字段+100;

update 数据表2 set status=1;

}