一、 准备文件
文档资料
根据 客户提供文档《CHINAPAY商户控制台使用手册 》登陆 http://console.chinapay.com/newgms (需要客户提供 商户号,操作员号及密码)
下载
解压文件后得到如下文件
还需要2个key(MerPrK.key和PgPubk.key)文件,分别是商户私匙和公匙,需要客户签订合同后银联才提供。
二、安装部署
1. 将ChinaPay.dll、CPNPC.dll及netpay.dll复制bin下
2. 注册Chinapay.dll和CPNPC.dll(指令regsvr32 )
例如 regsvr32 F:\工作\MySolution\Web\bin\Chinapay.dll
regsvr32 F:\工作\MySolution\Web\bin\CPNPC.dll
3. Com类型信息转换为.NET元数据, 需要vs命令提示 工具下运行
tlbimp F:\工作\MySolution\Web\bin\Chinapay.dll /out: F:\工作\MySolution\Web\bin\ ChinaPay_loaf.dll
站点增加引用ChinaPay_loaf.dll 使用的时候引用using ChinaPay_loaf 即可
4. 写一个类Chinapay.cs放在App_Code下 用于 用户签名 和验证,不理解签名和验证的看《附3商户技术开发手册.doc》数字签名,
using System.Data;
using System.Configuration;
using System.Web;
using System.Web.Security;
using System.Web.UI;
using System.Web.UI.HtmlControls;
using System.Web.UI.WebControls;
using System.Web.UI.WebControls.WebParts;
using ChinaPay_loaf ;
using System.Web.UI;
/// <summary>
///Chinapay 的摘要说明
/// </summary>
public class Chinapay
{
string strUrl =HttpContext.Current.Request.PhysicalApplicationPath; //获取网站根目录物理路径
public Chinapay()
{
}
/// <summary>
/// 订单签名函数sign
/// </summary>
/// <param name="MerId">商户号,长度为15个字节的数字串,由ChinaPay或清算银行分配</param>
/// <param name="OrdId">订单号,长度为16个字节的数字串,由用户系统/网站生成,失败的订单号允许重复支付</param>
/// <param name="TransAmt">交易金额,长度为12个字节的数字串,例如:数字串"000000001234"表示12.34元</param>
/// <param name="CuryId">货币代码, 长度为3个字节的数字串,目前只支持人民币,取值为"156"</param>
/// <param name="TransDate">交易日期,长度为8个字节的数字串,表示格式为:YYYYMMDD</param>
/// <param name="TransType">交易类型,长度为4个字节的数字串,取值范围为:"0001"和"0002", 其中"0001"表示消费交易,"0002"表示退货交易</param>
/// <returns>string CheckValue[256] 即NetPayClient根据上述输入参数生成的商户数字签名,长度为256字节的字符串</returns>
public string getSign(string MerId, string OrdId, string TransAmt, string CuryId, string TransDate, string TransType)
{
NetPayClientClass npc = new NetPayClientClass(); //实例NetPay签名
string temp = strUrl + "key\\MerPrK.key";
npc.setMerKeyFile(strUrl + "\\App_Data\\MerPrK.key");
string strChkValue = ""; //chinapay返回的商户数字签名
strChkValue = npc.sign(MerId, OrdId, TransAmt, CuryId, TransDate, TransType);
return strChkValue.Trim();
}
/// <summary>
/// 对一段字符进行签名 signData
/// </summary>
/// <param name="MerId">商户号,长度为15个字节的数字串,由ChinaPay分配</param>
/// <param name="SignMsg">用于要签名的字符串</param>
/// <returns>String CheckValue[256]即NetPayClient根据上述输入参数生成的商户数字签名,长度为256字节的字符串</returns>
public string signData(string MerId, string SignMsg)
{
NetPayClientClass npc = new NetPayClientClass(); //实例NetPay签名
npc.setMerKeyFile(strUrl + "App_Data\\MerPrK.key");
string strChkValueData = "";
strChkValueData = npc.signData(MerId, SignMsg);
return strChkValueData.Trim();
}
/// <summary>
/// 验证交易应答函数check
/// </summary>
/// <param name="MerId">商户号,长度为15个字节的数字串,由ChinaPay分配</param>
/// <param name="OrdId">订单号,长度为16个字节的数字串,由商户系统生成,失败的订单号允许重复支付</param>
/// <param name="TransAmt">交易金额,长度为12个字节的数字串,例如:数字串"000000001234"表示12.34元</param>
/// <param name="CuryId">货币代码, 长度为3个字节的数字串,目前只支持人民币,取值为"156"</param>
/// <param name="TransDate">交易日期,长度为8个字节的数字串,表示格式为: YYYYMMDD</param>
/// <param name="TransType">交易类型,长度为4个字节的数字串,取值范围为:"0001"和"0002", 其中"0001"表示消费交易,"0002"表示退货交易</param>
/// <param name="OrderStatus">交易状态,长度为4个字节的数字串。详见交易状态码说明</param>
/// <param name="CheckValue">校验值,即ChinaPay对交易应答的数字签名,长度为256字节的字符串</param>
/// <returns>true 表示成功,即该交易应答为ChinaPay所发送,商户根据“交易状态”进行后续处理;否则表示失败,即无效应答,商户可忽略该应答</returns>
public bool getCheck(string MerId, string OrdId, string TransAmt, string CuryId, string TransDate, string TransType, string OrderStatus, string CheckValue)
{
NetPayClientClass npc = new NetPayClientClass(); //实例NetPay签名
npc.setPubKeyFile(strUrl + "App_Data\\PgPubk.key");
string strFlag = "";
bool bolFlag = false;
strFlag = npc.check(MerId, OrdId, TransAmt, CuryId, TransDate, TransType, OrderStatus, CheckValue); // ChkValue 为ChinaPay返回给商户的域段内容
if (strFlag == "0") //“0”表示验签成功
bolFlag = true;
return bolFlag;
}
/// <summary>
/// 对一段字符串进行签名验证 checkData
/// </summary>
/// <param name="PlainData">用于数字签名的字符串</param>
/// <param name="CheckValue">校验值,要验证的字符串的数字签名,长度为256字节的字符串</param>
/// <returns>true 表示验证通过成功;否则表示失败</returns>
public bool checkData(string PlainData, string CheckValue)
{
NetPayClientClass npc = new NetPayClientClass(); //实例NetPay签名
npc.setPubKeyFile(strUrl + "App_Data\\PgPubk.key");
string strFlagData = "";
bool bolFlagData = false;
strFlagData = npc.checkData(PlainData, CheckValue);
if (strFlagData == "true")
bolFlagData = true;
return bolFlagData;
}
//
//支付函数
/// <summary>
/// 支付函数
/// </summary>
/// <param name="OrderID">程序 订单编号</param>
/// <param name="TransAmt">交易钱数</param>
/// <param name="proName">产品名称 可选</param>
public void GoToPay(string OrderID, string TransAmt, string proName,string gateid)
{
Chinapay cpy = new Chinapay();
//获取传递给银联chinapay的各个参数-----------------------------------------------
// string cpyUrl = "http://payment-test.chinapay.com/pay/TransGet"; //测试地址,测试的时候用这个地址,应用到网站时用下面那个地址
string cpyUrl = "http://payment.chinapay.com/pay/TransGet";
string cpyMerId = "808080580112345"; //ChinaPay统一分配给商户的商户号,15位长度,必填
string cpyOrdId = getOrderID(OrderID); //商户提交给ChinaPay的交易订单号,订单号的第五至第九位必须是商户号的最后五位,即“12345”;16位长度,必填
string cpyTransAmt = getTransAmt(TransAmt); //订单交易金额,12位长度,左补0,必填,单位为分,000000000001 表示 12.34 元
string cpyCuryId = "156"; //订单交易币种,3位长度,固定为人民币156,必填
string cpyTransDate = DateTime.Now.ToString("yyyyMMdd"); //订单交易日期,8位长度,必填,格式yyyyMMdd
string cpyTransType = "0001"; //交易类型,4位长度,必填,0001表示消费交易,0002表示退货交易
string cpyVersion = "20040916"; //支付接入版本号,808080开头的商户用此版本,必填,另一版本为"20070129"
string cpyBgRetUrl = "http://test003.abc.cc/Chinapay_Bgreturn.aspx"; //后台交易接收URL,为后台接受应答地址,用于商户记录交易信息和处理,对于使用者是不可见的,长度不要超过80个字节,必填
string cpyPageRetUrl = "http://test003.abc.cc/Chinapay_Pgreturn.aspx"; //页面交易接收URL,为页面接受应答地址,用于引导使用者返回支付后的商户网站页面,长度不要超过80个字节,必填
string cpyGateId = gateid; //支付网关号,可选,参看银联网关类型,如填写GateId(支付网关号),则消费者将直接进入支付页面,否则进入网关选择页面,可登陆商户管理平台 查看各个银行的网管号
string cpyPriv1 = proName; //商户私有域,长度不要超过60个字节,商户通过此字段向Chinapay发送的信息,Chinapay依原样填充返回给商户
string strChkValue = ""; //256字节长的ASCII码,此次交易所提交的关键数据的数字签名,必填
strChkValue = cpy.getSign(cpyMerId, cpyOrdId, cpyTransAmt, cpyCuryId, cpyTransDate, cpyTransType);
if (strChkValue != "")
{
HttpContext.Current.Response.Write("<form name='chinapayForm' method='post' action='" + cpyUrl + "'>"); //支付地址
HttpContext.Current.Response.Write("<input type='hidden' name='MerId' value='" + cpyMerId + "' />"); //商户号
HttpContext.Current.Response.Write("<input type='hidden' name='OrdId' value='" + cpyOrdId + "' />"); //订单号
HttpContext.Current.Response.Write("<input type='hidden' name='TransAmt' value='" + cpyTransAmt + "' />"); //支付金额
HttpContext.Current.Response.Write("<input type='hidden' name='CuryId' value='" + cpyCuryId + "' />"); //交易币种
HttpContext.Current.Response.Write("<input type='hidden' name='TransDate' value='" + cpyTransDate + "' />"); //交易日期
HttpContext.Current.Response.Write("<input type='hidden' name='TransType' value='" + cpyTransType + "' />"); //交易类型
HttpContext.Current.Response.Write("<input type='hidden' name='Version' value='" + cpyVersion + "' />"); //支付接入版本号
HttpContext.Current.Response.Write("<input type='hidden' name='BgRetUrl' value='" + cpyBgRetUrl + "' />"); //后台接受应答地址
HttpContext.Current.Response.Write("<input type='hidden' name='PageRetUrl' value='" + cpyPageRetUrl + "' />"); //为页面接受应答地址
HttpContext.Current.Response.Write("<input type='hidden' name='GateId' value='" + cpyGateId + "' />"); //支付网关号
HttpContext.Current.Response.Write("<input type='hidden' name='Priv1' value='" + cpyPriv1 + "' />"); //商户私有域,这里将订单自增编号放进去了
HttpContext.Current.Response.Write("<input type='hidden' name='ChkValue' value='" + strChkValue + "' />"); //此次交易所提交的关键数据的数字签名
HttpContext.Current.Response.Write("<script>");
HttpContext.Current.Response.Write("document.chinapayForm.submit();");
HttpContext.Current.Response.Write("</script></form>");
}
}
//订单号
private string getOrderID(string orderID)
{
//程序中的订单号 案例12022800001
string orderid = string.Format("{0}{1}{2}", orderID.Substring(0, 4), "12345", orderID.Substring(4)); //订单前四位+商家最后五位+订单后七位
return orderid;
}
//返回交易金额
private string getTransAmt(string count)
{
string moneyCount = count.ToString().Replace(".", "");
return moneyCount.PadLeft(12, '0');
}
}
5. 下面的需要解决的就是写四个页面(支付页,银行选择页面,后台接受处理页和用户支付成功后跳转页),银行选择页面,主要是自己定义图标,这里目的就是为了选择不同的网关来对应不同的银行.罗列出常用的银行。
{
Chinapay cpy = new Chinapay();
string TransDate = "",MerId = "",OrdId = "",TransType = "",TransAmt = "",CuryId = "",ChkValue = "",OrderStatus = "",GateId = "",Priv1 = "";
bool bolCheck=false;
TransDate = Request["transdate"].Trim(); //交易日期
MerId = Request["merid"].Trim(); //商家号
OrdId = Request["orderno"].Trim(); //订单号
TransType = Request["transtype"].Trim();//交易类型
TransAmt = Request["amount"].Trim(); //交易货币值
CuryId = Request["currencycode"].Trim(); //交易币种
ChkValue = Request["checkvalue"].Trim();
OrderStatus = Request["status"].Trim(); //订单状态
GateId = Request["GateId"].Trim(); //支付网关号
Priv1 = Request["Priv1"].Trim(); //商户私有域
///检验是否是银联chinapay返回的交易数据
bolCheck = cpy.getCheck(MerId,OrdId,TransAmt,CuryId,TransDate,TransType,OrderStatus,ChkValue);
if (bolCheck){
if (OrderStatus == "1001")//交易成功
{
string myOrderID = OrdId.Replace("12345", string.Empty);//移除商户号后正好是我系统的订单号
//更新订单表
OrderDetailBLL bll = new OrderDetailBLL();
bll.UpdateStateAll(myOrderID);//订单号
}
}
}
三、需要注意的地方
1. 后台接受页面验证签名后还要判断下订单状态 只有1001状态才是成功交易,退款其他 要看文档
2. 商户提交给ChinaPay的交易订单号,订单号的第五至第九位必须是商户号的最后五位(04版的,现在07版不用这样做了),即“12345”;16位长度
例如商家号是: 8080805801123456 则提交的订单号必须如下形式
例如 2012123450000000 ;
这里你要看着 怎么处理比较好,根据这个订单号 能对应上,你网站里面的订单号
我的订单号生成规则是
订单号=年数后2位+月数2位+天2位+五位自增数(不够补0);
例如 :12030100001; //12年3月1号 第00001个单子,能看出当天的销售量 也就是说 每天售量最大值是99999,一般小商场足够用了。