微信支付流程梳理
1.小程序微信支付
/**
* 支付接口
* @param request
* @param response
* @throws Exception
*/
@SuppressWarnings("rawtypes")
@RequestMapping(value="/payOrderCom",produces = MediaType.APPLICATION_JSON_VALUE + ";charset=UTF-8")
@ResponseBody
public String payOrderCom(HttpServletRequest request, HttpServletResponse response) throws Exception{
Map<String, Object> map=new HashMap<String, Object>();
request.setCharacterEncoding("UTF-8");
response.setCharacterEncoding("UTF-8");
//小程序appid
String appid = request.getParameter("appid");
//商家id
String mch_id = request.getParameter("mch_id");
//订单id
String indent_id = request.getParameter("indent_id");
/*String bitsecret_key = request.getParameter("bitsecret_key"); //32位秘钥*/
if(!isNull(appid) || !isNull(mch_id) ||!isNull(indent_id)){
map.put("type", false);
map.put("massage", "未获取到参数");
return JsonMapper.getInstance().toJson(map);
}
Indent indent = indentService.findOneByIndent(indent_id);
if(indent == null){
map.put("type", false);
map.put("massage", "未获取订单信息");
return JsonMapper.getInstance().toJson(map);
}
//得到价钱(自定义)
String price = indent.getTotalMoney();
BigDecimal fee = BigDecimal.valueOf(Long.valueOf(price)).divide(new BigDecimal(100));
//订单标题(自定义)
String title = request.getParameter("title");
//时间戳
String times = System.currentTimeMillis() + "";
//获取客户端的ip
String spbill_create_ip = XMLUtil.getIpAddr(request);
//订单编号(自定义 这里以时间戳+随机数)
SortedMap<Object, Object> packageParams = new TreeMap<Object, Object>();
packageParams.put("appid", appid);//微信小程序ID
packageParams.put("mch_id", mch_id);//商户ID
packageParams.put("nonce_str", times);//随机字符串(32位以内) 这里使用时间戳
packageParams.put("body", title);//支付主体名称 自定义
packageParams.put("out_trade_no", indent_id);//编号 自定义以时间戳+随机数+商品ID(订单id)
packageParams.put("total_fee", fee);//价格 自定义
packageParams.put("spbill_create_ip", spbill_create_ip);//获取客户端ip
packageParams.put("notify_url", "http://192.168.1.108/mkkMoblie/payOrder/buyOrder");//支付返回地址要外网访问的到, localhost不行,调用下面buy方法。(订单存入数据库)
packageParams.put("trade_type", "JSAPI");//这个api有,固定的
//获取sign
String sign = PayCommonUtil.createSign("UTF-8", packageParams);//最后这个是自己在微信商户设置的32位密钥
packageParams.put("sign", sign);
//转成XML
String requestXML = PayCommonUtil.getRequestXml(packageParams);
System.out.println(requestXML);
//得到含有prepay_id的XML
String resXml = HttpUtil.postData("https://api.mch.weixin.qq.com/pay/unifiedorder", requestXML);
Map map_pay = XMLUtil.doXMLParse(resXml);
System.out.println(map_pay);
// String return_code = (String) map.get("return_code");
//得到prepay_id
String prepay_id = (String) map_pay.get("prepay_id");
SortedMap<Object, Object> packageP = new TreeMap<Object, Object>();
packageP.put("appId", appid);//!!!注意,这里是appId,上面是appid
packageP.put("nonceStr", times);//时间戳
packageP.put("package", "prepay_id=" + prepay_id);//必须把package写成 "prepay_id="+prepay_id这种形式,两小时有效
packageP.put("signType", "MD5");//paySign加密
packageP.put("timeStamp", (System.currentTimeMillis() / 1000) + "");
//得到paySign
String paySign = PayCommonUtil.createSign("UTF-8", packageP);
packageP.put("paySign", paySign);
//将packageP数据返回给小程序
/*Gson gson = new Gson();
String json = gson.toJson(packageP);
PrintWriter pw = response.getWriter();
System.out.println(json);
pw.write(json);
pw.close();*/
map.put("packageP", packageP);
map.put("type", true);
map.put("massage", "查询成功");
return JsonMapper.getInstance().toJson(map);
}
/**
* 支付回调
* @param request
* @param response
* @return
* @throws Exception
*/
@RequestMapping(value="/buyOrder",produces = MediaType.APPLICATION_JSON_VALUE + ";charset=UTF-8")
@ResponseBody
public String buyOrder(HttpServletRequest request,HttpServletResponse response) throws Exception{
Map<String, Object> map=new HashMap<String, Object>();
BufferedReader br = new BufferedReader(new InputStreamReader((ServletInputStream)request.getInputStream()));
String line = null;
StringBuilder sb = new StringBuilder();
if((line = br.readLine()) != null){
sb.append(line);
}
br.close();
//sb为微信返回的xml
String notityXml = sb.toString();
String resXml = "";
@SuppressWarnings("rawtypes")
Map map_xml = XMLUtil.doXMLParse(notityXml);
String returnCode = (String) map_xml.get("return_code");
System.out.println(map_xml);
if("SUCCESS".equals(returnCode)){
String out_trade_no=(String) map_xml.get("out_trade_no"); //商户订单号
Indent indent = indentService.findOneByIndent(out_trade_no);
if(indent == null){
map.put("type", false);
map.put("massage", "未获取订单信息");
return JsonMapper.getInstance().toJson(map);
}
indent.setIndentStatus("4");
indentService.update(indent);
resXml = "<xml>" + "<return_code><![CDATA[SUCCESS]]></return_code>"
+ "<return_msg><![CDATA[OK]]></return_msg>" + "</xml> ";
}else {
resXml = "<xml>" + "<return_code><![CDATA[FAIL]]></return_code>"
+ "<return_msg><![CDATA[报文为空]]></return_msg>" + "</xml> ";
}
map.put("resXml",resXml.getBytes());
map.put("type", true);
map.put("massage", "查询成功");
return JsonMapper.getInstance().toJson(map);
}
/**
用户支付回调
**/
@RequestMapping(value="/orderPay",produces = MediaType.APPLICATION_JSON_VALUE + ";charset=UTF-8")
@ResponseBody
public String orderPay(HttpServletRequest request, HttpServletResponse response) throws Exception {
Map<String, Object> map_return =new HashMap<String, Object>();
String resXml = "";
Map<String, String> backxml = new HashMap<String, String>();
InputStream inStream;
try {
inStream = request.getInputStream();
ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len = 0;
while ((len = inStream.read(buffer)) != -1) {
outSteam.write(buffer, 0, len);
}
//logger.error("微信支付----付款成功----");
outSteam.close();
inStream.close();
String result = new String(outSteam.toByteArray(), "utf-8");// 获取微信调用我们notify_url的返回信息
//logger.error("微信支付----result----=" + result);
//Map<Object, Object> map = Xmlunit.xml2map(result, false);
Map<Object, Object> map = XMLUtil.doXMLParse(result);
System.out.println(map);
if (map.get("result_code").toString().equalsIgnoreCase("SUCCESS")) {
//logger.error("微信支付----返回成功");
// ------------------------------
String order_num = map.get("out_trade_no").toString();
System.out.println("支付回调成功");
System.out.println(order_num);
// 此处的paySuccess为自己写的方法,主要作用是支付成功之后执行的一些代码,有需要自己写一个或者直接去掉均可。
paySuccess(order_num);
// 处理业务完毕 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// ------------------------------ 通过订单号 修改状态
request.getSession().setAttribute("message", "支付成功");
resXml = "<xml>" + "<return_code><![CDATA[SUCCESS]]></return_code>"
+ "<return_msg><![CDATA[OK]]></return_msg>" + "</xml> ";
BufferedOutputStream out = new BufferedOutputStream(response.getOutputStream());
out.write(resXml.getBytes());
out.flush();
out.close();
}
} catch (IOException e) {
e.printStackTrace();
}
map_return.put("resXml",resXml.getBytes());
map_return.put("type", true);
map_return.put("massage", "支付成功");
return JsonMapper.getInstance().toJson(map_return);
}