一,银联支付的整体流程
客户端提供服务器给服务器订单信息----服务器端拿到数据推送给银联指定的地址----银联给服务器端返回一个流水账号----服务器将流水账号返给客户端
----客户端由于集成了银联控件,根据流水账号调用银联支付界面进行支付----支付完成之后银联会回调服务器端提供的回调地址----服务器端根据回调参数处理业务逻辑
二,服务器端主要代码解析.
upmp.properties security.key=******//银联技术给我们提供的商户密钥 mer.id=88000000000***** //商户的银联账号 mer.back.end.url=/servlet/unionpaycallback//支付成功后银联的回调地址 mer.front.end.url=http://www.yourdomain.com/your_path/yourFrontEndUrl//支付成功后银联回调的前段地址,可不填 # message info version=1.0.0//版本号 charset=UTF-8//编码 sign.method=MD5//加密方法 # server url upmp.trade.url=http://222.66.233.198:8080/gateway/merchant/trade//处理交易请求的银联地址 upmp.query.url=http://222.66.233.198:8080/gateway/merchant/query//查询交易的银联地址 card=6226440123456785//银联提供的测试账户 password=111101//银联提供的测试密码 下面是给银联推送订单的代码. /** * 银联支付推送给银联 */ public String executeUnionPayMoney(String auth, String info, String basePath) { //returnMap--返回的Map //returnParams--定义的参数返回类型 Map<String,Object> returnMap = new HashMap<String,Object>(); String result =""; try{ //把info中的信息解析成map @SuppressWarnings("unchecked") Map<String,Object> map = JSON.parseObject(info, Map.class); OrderDao orderDao = (OrderDao) ac.getBean("orderDao"); if(map.get("order_id")!=null&&!map.get("order_id").equals("")&& map.get("user_id")!=null&&!map.get("user_id").equals("")&& map.get("payMoney")!=null&&!map.get("payMoney").equals("")){ //把订单数据推给银联(订单推送接口),根据返回结果是否为true Order order = orderDao.getById(Long.valueOf(map.get("order_id").toString())); Map<String, String> req = new HashMap<String, String>(); Map<String, String> reqReservedMap = new HashMap<String, String>(); reqReservedMap.put("user_id", map.get("user_id").toString()); reqReservedMap.put("order_id", map.get("order_id").toString()); reqReservedMap.put("opt", "0");//自己自定义的参数,把Map转换成json字符串,通过请求方保留域reqReserved传递给银联,方便银联回调服务器时,服务器处理业务逻辑使用 req.put("version", UpmpConfig.VERSION);// 版本号 req.put("charset", UpmpConfig.CHARSET);// 字符编码 req.put("transType", "01");// 交易类型 req.put("merId", UpmpConfig.MER_ID);// 商户代码 //req.put("backEndUrl",basePath.substring(0, basePath.length()-1) + UpmpConfig.MER_BACK_END_URL);// 通知URL req.put("backEndUrl",basePath.substring(0, basePath.length()-1) + UpmpConfig.MER_BACK_END_URL);// 通知URL 我自己提供给银联支付成功后的回调地址 req.put("frontEndUrl", UpmpConfig.MER_FRONT_END_URL);// 前台通知URL(可选) req.put("orderDescription", "航空机票付款");// 订单描述(可选) req.put("orderTime", new SimpleDateFormat("yyyyMMddHHmmss").format(new Date()));// 交易开始日期时间yyyyMMddHHmmss req.put("orderTimeout", "");// 订单超时时间yyyyMMddHHmmss(可选);默认的是一小时 req.put("orderNumber", orderDao.getById(Long.valueOf(map.get("order_id").toString())).getOrderNum());//订单号(商户根据自己需要生成订单号) req.put("orderAmount", Integer.valueOf(map.get("payMoney").toString())*100+"");// 订单金额;因为这里默认的单位是分,而通常客户端传过来的单位是元,所以需要再此 //处进行*100的操作,防止因为单位换算导致支付出现问题,需要注意的是:银联回调时你如果从回调的参数中拿settleAmount,记得进行除100的操作 req.put("orderCurrency", "156");// 交易币种(可选) 156代表人民币 req.put("reqReserved", new Gson().toJson(reqReservedMap));// 请求方保留域(可选,用于透传商户信息)//此处用于传递咱们自己的参数,可以再银联回调时使用, //此处我是以json字符串的方式传递参数的,之后解析json字符串即可获取相关参数. Map<String, String> resp = new HashMap<String, String>(); boolean validResp = UpmpService.trade(req, resp); if(order.getPayState()==0){ returnMap.put("errcode",-98); returnMap.put("msg","该订单已支付"); }else{ // 商户的业务逻辑 if (validResp){ // 服务器应答签名验证成功 returnMap.put("tn", resp.get("tn"));//交易流水号,默认的参数就是tn returnMap.put("errcode",0); returnMap.put("msg","成功返回流水号"); }else { // 服务器应答签名验证失败 returnMap.put("errcode",-100); returnMap.put("msg","系统出错"); } } }else { returnMap.put("errcode",-9); returnMap.put("msg","传值错误"); } }catch (Exception e) { e.printStackTrace(); returnMap.put("errcode",-99); returnMap.put("msg","服务器出现异常"); } result=JSON.toJSONString(returnMap); return result; }
CallBackUnionPay.java 这个是我自己写的一个银联回调的servlet,opt为自定义参数,opt=0代表支付订单,opt=1代表充值操作 /** * Alipay.com Inc. * Copyright (c) 2005-2008 All Rights Reserved. */ package com.unionpay.upmp.sdk.callback; import java.io.IOException; import java.io.PrintWriter; import java.util.Date; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import com.airplane.base.AbstractExecution; import com.airplane.dao.AccountDao; import com.airplane.dao.IntegralDao; import com.airplane.dao.OrderDao; import com.airplane.dao.UserDao; import com.airplane.domain.Account; import com.airplane.domain.Integral; import com.airplane.domain.Order; import com.airplane.domain.User; import com.google.gson.Gson; import com.unionpay.upmp.sdk.service.UpmpService; /** * 接收通知并处理 * * @author gxw */ public class CallBackUnionPay extends HttpServlet { private static final long serialVersionUID = 7216412938937049671L; private static Log logger = LogFactory.getLog(CallBackUnionPay.class); @SuppressWarnings("unchecked") public void doPost(HttpServletRequest request, HttpServletResponse response){ logger.info("==============================here==========================="); System.out.println("接收到通知!"); PrintWriter out; try { out = response.getWriter(); Map<String,String> params = new HashMap<String,String>(); Map requestParams = request.getParameterMap(); logger.info("request的参数:"+requestParams); for (Iterator iter = requestParams.keySet().iterator(); iter.hasNext();) { String name = (String) iter.next(); String[] values = (String[]) requestParams.get(name); String valueStr = ""; for (int i = 0; i < values.length; i++) { valueStr = (i == values.length - 1) ? valueStr + values[i] : valueStr + values[i] + ","; } params.put(name, valueStr); } logger.info("request处理后的参数:"+params); Map<String,String> personParams = new Gson().fromJson(params.get("reqReserved"), Map.class);//解析在上一步自定义的json参数,方便处理业务逻辑. if(UpmpService.verifySignature(params)){// 服务器签名验证成功 //请在这里加上商户的业务逻辑程序代码 if("0".equals(personParams.get("opt"))){ String user_id = personParams.get("user_id"); String order_id = personParams.get("order_id"); logger.info("user_id"+user_id+"&order_id="+order_id); //获取通知返回参数,可参考接口文档中通知参数列表(以下仅供参考) String transStatus = request.getParameter("transStatus");// 交易状态 if (null != transStatus && transStatus.equals("00")) { // 交易处理成功 } out.println("success"); } if("1".equals(personParams.get("opt"))){ String payMoney2 = params.get("settleAmount"); String user_id2 = personParams.get("user_id"); //获取通知返回参数,可参考接口文档中通知参数列表(以下仅供参考) String transStatus2 = request.getParameter("transStatus");// 交易状态 if (null != transStatus2 && transStatus2.equals("00")) { // 交易处理成功 } out.println("success"); } else{// 服务器签名验证失败 out.println("fail"); } } }catch (IOException e) { e.printStackTrace(); logger.error(e); } } /*public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException { doPost(request, response); }*/ }