無圖無真相,所以先上一波圖
做這個項目之前,你必須要有p1_MerId和keyValue,這個需要自己去官網申請的,這裏我提供做測試的。p1_MerId="10001126856"和keyValue="69cl522AV6q613Ii4W6u8K6XuW8vM1N6bFgyv769220IuYe9u37N4y7rI4Pl";
然後需要來個工具類,一個是用來加密的MD5加密算法,一個是用來對關鍵屬性進行加密的工具類。具體代碼如下所示:
1.DigestUtil.java
package com.bd.utils;
import java.io.UnsupportedEncodingException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;
public class DigestUtil {
private static String encodingCharset = "UTF-8";
/**
* @param aValue
* @param aKey
* @return
*/
public static String hmacSign(String aValue, String aKey) {
byte k_ipad[] = new byte[64];
byte k_opad[] = new byte[64];
byte keyb[];
byte value[];
try {
keyb = aKey.getBytes(encodingCharset);
value = aValue.getBytes(encodingCharset);
} catch (UnsupportedEncodingException e) {
keyb = aKey.getBytes();
value = aValue.getBytes();
}
Arrays.fill(k_ipad, keyb.length, 64, (byte) 54);
Arrays.fill(k_opad, keyb.length, 64, (byte) 92);
for (int i = 0; i < keyb.length; i++) {
k_ipad[i] = (byte) (keyb[i] ^ 0x36);
k_opad[i] = (byte) (keyb[i] ^ 0x5c);
}
MessageDigest md = null;
try {
md = MessageDigest.getInstance("MD5");
} catch (NoSuchAlgorithmException e) {
return null;
}
md.update(k_ipad);
md.update(value);
byte dg[] = md.digest();
md.reset();
md.update(k_opad);
md.update(dg, 0, 16);
dg = md.digest();
return toHex(dg);
}
public static String toHex(byte input[]) {
if (input == null)
return null;
StringBuffer output = new StringBuffer(input.length * 2);
for (int i = 0; i < input.length; i++) {
int current = input[i] & 0xff;
if (current < 16)
output.append("0");
output.append(Integer.toString(current, 16));
}
return output.toString();
}
/**
*
* @param args
* @param key
* @return
*/
public static String getHmac(String[] args, String key) {
if (args == null || args.length == 0) {
return (null);
}
StringBuffer str = new StringBuffer();
for (int i = 0; i < args.length; i++) {
str.append(args[i]);
}
return (hmacSign(str.toString(), key));
}
/**
* @param aValue
* @return
*/
public static String digest(String aValue) {
aValue = aValue.trim();
byte value[];
try {
value = aValue.getBytes(encodingCharset);
} catch (UnsupportedEncodingException e) {
value = aValue.getBytes();
}
MessageDigest md = null;
try {
md = MessageDigest.getInstance("SHA");
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
return null;
}
return toHex(md.digest(value));
}
}
2.PanymentUtil.java
package com.bd.utils;
public class PanymentUtil {
/**
* 生成hmac方法
*
* @param p0_Cmd 業務類型
* @param p1_MerId 商戶編號
* @param p2_Order 商戶訂單號
* @param p3_Amt 支付金額
* @param p4_Cur 交易幣種
* @param p5_Pid 商品名稱
* @param p6_Pcat 商品種類
* @param p7_Pdesc 商品描述
* @param p8_Url 商戶接收支付成功數據的地址
* @param p9_SAF 送貨地址
* @param pa_MP 商戶擴展信息
* @param pd_FrpId 銀行編碼
* @param pr_NeedResponse 應答機制
* @param keyValue 商戶密鑰
* @return
*/
public static String buildHmac(String p0_Cmd,String p1_MerId,
String p2_Order, String p3_Amt, String p4_Cur,String p5_Pid, String p6_Pcat,
String p7_Pdesc,String p8_Url, String p9_SAF,String pa_MP,String pd_FrpId,
String pr_NeedResponse,String keyValue) {
StringBuffer sValue = new StringBuffer();
// 業務類型
sValue.append(p0_Cmd);
// 商戶編號
sValue.append(p1_MerId);
// 商戶訂單號
sValue.append(p2_Order);
// 支付金額
sValue.append(p3_Amt);
// 交易幣種
sValue.append(p4_Cur);
// 商品名稱
sValue.append(p5_Pid);
// 商品種類
sValue.append(p6_Pcat);
// 商品描述
sValue.append(p7_Pdesc);
// 商戶接收支付成功數據的地址
sValue.append(p8_Url);
// 送貨地址
sValue.append(p9_SAF);
// 商戶擴展信息
sValue.append(pa_MP);
// 銀行編碼
sValue.append(pd_FrpId);
// 應答機制
sValue.append(pr_NeedResponse);
String sNewString = DigestUtil.hmacSign(sValue.toString(), keyValue);
return sNewString;
}
/**
* 返回校驗hmac方法
*
* @param hmac 支付網關發來的加密驗證碼
* @param p1_MerId 商戶編號
* @param r0_Cmd 業務類型
* @param r1_Code 支付結果
* @param r2_TrxId 易寶支付交易流水號
* @param r3_Amt 支付金額
* @param r4_Cur 交易幣種
* @param r5_Pid 商品名稱
* @param r6_Order 商戶訂單號
* @param r7_Uid 易寶支付會員ID
* @param r8_MP 商戶擴展信息
* @param r9_BType 交易結果返回類型
* @param keyValue 密鑰
* @return
*/
public static boolean verifyCallback(String hmac, String p1_MerId,
String r0_Cmd, String r1_Code, String r2_TrxId, String r3_Amt,
String r4_Cur, String r5_Pid, String r6_Order, String r7_Uid,
String r8_MP, String r9_BType, String keyValue) {
StringBuffer sValue = new StringBuffer();
// 商戶編號
sValue.append(p1_MerId);
// 業務類型
sValue.append(r0_Cmd);
// 支付結果
sValue.append(r1_Code);
// 易寶支付交易流水號
sValue.append(r2_TrxId);
// 支付金額
sValue.append(r3_Amt);
// 交易幣種
sValue.append(r4_Cur);
// 商品名稱
sValue.append(r5_Pid);
// 商戶訂單號
sValue.append(r6_Order);
// 易寶支付會員ID
sValue.append(r7_Uid);
// 商戶擴展信息
sValue.append(r8_MP);
// 交易結果返回類型
sValue.append(r9_BType);
String sNewString = DigestUtil.hmacSign(sValue.toString(), keyValue);
if (hmac.equals(sNewString)) {
return true;
}
return false;
}
}
3.支付界面爲了具體簡化我做了一個簡單的銀行選擇界面
<%@ page language="java" contentType="text/html; charset=GBK"
%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<form action="PaymentRequest.do" method="post">
<input type="radio" name="jianshe123" value="jianshe1"/>建設銀行
<input type="submit" value="提交"/>
</form>
</body>
</html>
在這裏我順便附上易寶支付各大銀行的編碼
可直連銀行 | |
pd_FrpId參數值 | 對應支付通道名稱 |
1000000-NET | 易寶會員支付 |
ICBC-NET-B2C | 工商銀行 |
CMBCHINA-NET-B2C | 招商銀行 |
ABC-NET-B2C | 中國農業銀行 |
CCB-NET-B2C | 建設銀行 |
BCCB-NET-B2C | 北京銀行 |
BOCO-NET-B2C | 交通銀行 |
CIB-NET-B2C | 興業銀行 |
NJCB-NET-B2C | 南京銀行 |
CMBC-NET-B2C | 中國民生銀行 |
CEB-NET-B2C | 光大銀行 |
BOC-NET-B2C | 中國銀行 |
PINGANBANK-NET | 平安銀行 |
CBHB-NET-B2C | 渤海銀行 |
HKBEA-NET-B2C | 東亞銀行 |
NBCB-NET-B2C | 寧波銀行 |
ECITIC-NET-B2C | 中信銀行(需要證書才能連接到銀行) |
SDB-NET-B2C | 深圳發展銀行 |
GDB-NET-B2C | 廣發銀行 |
SHB-NET-B2C | 上海銀行 |
SPDB-NET-B2C | 上海浦東發展銀行 |
POST-NET-B2C | 中國郵政 |
BJRCB-NET-B2C | 北京農村商業銀行 |
HXB-NET-B2C | 華夏銀行(此功能默認不開通,如需開通請與易寶支付銷售人員聯繫) |
CZ-NET-B2C | 浙商銀行 |
HZBANK-NET-B2C | 杭州銀行 |
WAP銀行(支持WAP的支付通道的接入) | |
ICBC-WAP | 工商銀行WAP |
CMBCHINA-WAP | 招商銀行WAP |
CCB-WAP | 建設銀行WAP |
預付費卡網關(需要聯繫易寶支付銷售開通才可使用) | |
EBC-NET | 商聯通卡 |
AllScore-NET | 奧斯卡 |
Edenred-NET | 雅高e卡 |
Ybt-NET | 中欣銀寶通卡 |
Yikahui-NET | 壹卡會 |
Bohaiyisheng-NET | 易生如意卡 |
EPOS網關(需要聯繫易寶支付銷售開通才可使用) | |
EPOS-NET | 網關 |
快捷支付(需要聯繫易寶支付銷售開通才可使用)d | |
FASTPAY | 快捷支付網關 |
ABCCREDIT_FASTPAY | 農行快捷支付 |
ICBCCREDIT_FASTPAY | 工行快捷支付 |
BOCCREDIT_FASTPAY | 中行快捷支付 |
CCBCREDIT_FASTPAY | 建行快捷支付 |
非銀行卡Logo及詳細信息,請見http://www.yeepay.com/html/huodong/feiyin/ | |
可直連的其他支付通道(以下均需要聯繫易寶支付銷售開通才可使用) | |
JUNNET-NET | 駿網一卡通 |
SNDACARD-NET | 盛大卡 |
SZX-NET | 神州行 |
ZHENGTU-NET | 征途卡 |
QQCARD-NET | Q幣卡 |
UNICOM-NET | 聯通卡 |
JIUYOU-NET | 久遊卡 |
YPCARD-NET | 易寶e卡通 |
NETEASE-NET | 網易卡 |
WANMEI-NET | 完美卡 |
SOHU-NET | 搜狐卡 |
TELECOM-NET | 電信卡 |
ZONGYOU-NET | 縱遊一卡通 |
TIANXIA-NET | 天下一卡通 |
TIANHONG-NET | 天宏一卡通 |
BESTPAY-NET | 翼支付 |
PaymentRequest.java
package com.bd.servlet;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.bd.utils.PanymentUtil;
@WebServlet("/PaymentRequest.do")
public class PaymentRequest extends HttpServlet {
private static final long serialVersionUID = 1L;
public PaymentRequest() {
super();
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.getRequestDispatcher("/WEB-INF/payment.jsp").forward(request, response);
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.setCharacterEncoding("GBK");
String orid="11111111113";
String amout="0.01";
String pd_FrpId="CCB-NET-B2C";
String p1_MerId="10001126856";
String keyValue="69cl522AV6q613Ii4W6u8K6XuW8vM1N6bFgyv769220IuYe9u37N4y7rI4Pl";
String merchantCallbackURL="http://172.16.19.50:8080/payment/BackServlet.do";
String messageType="Buy";
String currency="CNY";
String productDesc="";
String productCat="";
String productId="";
String addressFlg="0";
String sMctProperties="";
String pr_NeedResponse="0";
String md5hmac=PanymentUtil.buildHmac(messageType, p1_MerId, orid, amout, currency, productId, productCat, productDesc, merchantCallbackURL, addressFlg, sMctProperties, pd_FrpId, pr_NeedResponse, keyValue);
// request.setCharacterEncoding("utf-8");
// response.setContentType("text/html;charset=utf-8");
// String p0_Cmd="Buy",
// p1_MerId="10001126856",
// p2_Order="11111111111",
// p3_Amt="0.01",
// p4_Cur="CNY",
// p5_Pid="",
// p6_Pcat="",
// p7_Pdesc="",
// p8_Url="http://localhost:8080/payment/backServlet.do",
// p9_SAF="",
// pa_MP="",
// pd_FrpId="BOC-NET-B2C",
// pr_NeedResponse="1";
// String keyValue="69cl522AV6q613Ii4W6u8K6XuW8vM1N6bFgyv769220IuYe9u37N4y7rI4Pl";
// String hmac=PanymentUtil.buildHmac(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, keyValue);
// String url="https://www.yeepay.com/app-merchant-proxy/node?"+
// "&p0_Cmd="+p0_Cmd+
// "&p1_MerId="+p1_MerId+
// "&p2_Order="+p2_Order+
// "&p3_Amt="+p3_Amt+
// "&p4_Cur="+p4_Cur+
// "&p5_Pid="+p5_Pid+
// "&p6_Pcat="+p6_Pcat+
// "&p7_Pdesc="+p7_Pdesc+
// "&p8_Url="+p8_Url+
// "&p9_SAF="+p9_SAF+
// "&pa_MP="+pa_MP+
// "&pd_FrpId="+pd_FrpId+
// "&pr_NeedResponse="+pr_NeedResponse+
// "&hmac="+hmac;
// response.sendRedirect(url);
request.setAttribute("businessType", messageType);
request.setAttribute("accountID", p1_MerId);
request.setAttribute("orderID", orid);
request.setAttribute("amount", amout);
request.setAttribute("currency", currency);
request.setAttribute("productID", productId);
request.setAttribute("productCategory", productCat);
request.setAttribute("productDesc", productDesc);
request.setAttribute("accountCallbackURL", merchantCallbackURL);
request.setAttribute("addressFlag", addressFlg);
request.setAttribute("accountMoreInfo", sMctProperties);
request.setAttribute("accountBankID", pd_FrpId);
request.setAttribute("needResponse", pr_NeedResponse);
request.setAttribute("md5hmac", md5hmac);
request.getRequestDispatcher("/WEB-INF/formrequset.jsp").forward(request, response);
}
}
我具體的測試接口網址我沒有找到,我找到的是真實的接口
https://www.yeepay.com/app-merchant-proxy/node
接下來我給大家看的一個表單界面,post請求到所給的網址
<%@ page language="java" contentType="text/html; charset=UTF-8"
%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body οnlοad="javascript:document.forms[0].submit()">
<form action="https://www.yeepay.com/app-merchant-proxy/node" method="POST" name="yeepay">
<!-- 以下hidden中的name值爲易寶支付規範的固定命名和順序 -->
<input type='hidden' name='p0_Cmd' value="${businessType}">
<input type='hidden' name='p1_MerId' value="${accountID}">
<input type='hidden' name='p2_Order' value="${orderID}">
<input type='hidden' name='p3_Amt' value="${amount}">
<input type='hidden' name='p4_Cur' value="${currency}">
<input type='hidden' name='p5_Pid' value="${productID}">
<input type='hidden' name='p6_Pcat' value="${productCategory}">
<input type='hidden' name='p7_Pdesc' value="${productDesc}">
<input type='hidden' name='p8_Url' value="${accountCallbackURL}">
<input type='hidden' name='p9_SAF' value="${addressFlag}">
<input type='hidden' name='pa_MP' value="${accountMoreInfo}">
<input type='hidden' name='pd_FrpId' value="${accountBankID}">
<input type="hidden" name='pr_NeedResponse' value="${needResponse}">
<input type='hidden' name='hmac' value="${md5hmac}">
</form>
</body>
</html>
接下來我們就按照具體的流程進行,付款完成之後會跳轉我們給的網址,即PaymentRequest.java的http://172.16.19.50:8080/payment/BackServlet.do
返回的請求該如何處理我也寫了一個servlet
BackServlet.java
package com.bd.servlet;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.oracle.jrockit.jfr.RequestableEvent;
@WebServlet("/BackServlet.do")
public class BackServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doPost(request, response);
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
request.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
String r1_Code=request.getParameter("r1_Code");
PrintWriter pw=response.getWriter();
if("1".equals(r1_Code))
{
String p1_MerId=request.getParameter("p1_MerId");
String r3_Amt=request.getParameter("r3_Amt");
String r6_Order=request.getParameter("r6_Order");
String rp_PayDate=request.getParameter("rp_PayDate");
request.setAttribute("msg", "支付成功!");
request.setAttribute("p1_MerId",p1_MerId );
request.setAttribute("r3_Amt", r3_Amt);
request.setAttribute("rp_PayDate", rp_PayDate);
request.getRequestDispatcher("/WEB-INF/paymentResult.jsp").forward(request, response);
// pw.println("支付成功!<br/>"
// + "商戶編號:"+p1_MerId+"<br/>"
// + "支付金額:"+r3_Amt+"<br/>"
// +"商戶訂單號:"+r6_Order+"<br/>"
// +"支付成功時間:"+rp_PayDate);
}
else
{
// pw.println("支付失敗!");
request.setAttribute("msg", "支付失敗!");
request.getRequestDispatcher("/WEB-INF/paymentResult.jsp").forward(request, response);
}
}
}
然後寫一個jsp具體接收
<%@
page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
<script type="text/javascript" src="dist/js/jquery-3.1.1.min.js" ></script>
<script type="text/javascript" src="dist/js/jquery.min.js" ></script>
<style type="text/css">
h1{
text-align:center;
}
</style>
<script type="text/javascript">
$(function(){
if($('h1').text()=='支付失敗!'){
$('h1').attr('style','color:red');
}else{
$('h1').attr('style','color:green');
}
})
</script>
</head>
<body>
<div style="margin:200px auto;width:300px">
<h1>${msg}</h1>
<h3>商戶編號:${p1_MerId}</h3>
<h3>支付金額:${r3_Amt }</h3>
<h3>商戶訂單號:${r6_Order}</h3>
<h3>支付成功時間:${rp_PayDate}</h3>
</div>
</body>
</html>