第三方支付易寶支付的具體實現

無圖無真相,所以先上一波圖


做這個項目之前,你必須要有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 翼支付
接下來是支付請求的servlet

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.javahttp://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>

發佈了36 篇原創文章 · 獲贊 34 · 訪問量 6萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章