org.redkalex.pay.WeiXinPayService Maven / Gradle / Ivy
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package org.redkalex.pay;
import java.io.*;
import java.security.*;
import java.util.*;
import java.util.concurrent.CompletableFuture;
import java.util.logging.*;
import java.util.regex.*;
import javax.annotation.Resource;
import javax.net.ssl.*;
import org.redkale.convert.json.JsonConvert;
import org.redkale.service.*;
import org.redkale.util.*;
import static org.redkalex.pay.PayRetCodes.*;
import static org.redkalex.pay.Pays.*;
/**
*
* 详情见: https://redkale.org
*
* @author zhangjx
*/
@Local
@AutoLoad(false)
public class WeiXinPayService extends AbstractPayService {
protected static final String format = "%1$tY%1$tm%1$td%1$tH%1$tM%1$tS"; //yyyyMMddHHmmss
protected static final Pattern PAYXML = Pattern.compile("<([^/>]+)>(.+)"); // "<([^/>]+)>"
//配置集合
protected Map elements = new HashMap<>();
@Resource(name = "property.pay.weixin.conf") //支付配置文件路径
protected String conf = "config.properties";
@Resource(name = "APP_HOME")
protected File home;
@Resource
protected JsonConvert convert;
@Override
public void init(AnyValue conf) {
if (this.convert == null) this.convert = JsonConvert.root();
this.reloadConfig(Pays.PAYTYPE_WEIXIN);
}
@Override
@Comment("判断是否支持指定支付类型")
public boolean supportPayType(final short paytype) {
return paytype == Pays.PAYTYPE_WEIXIN && !elements.isEmpty();
}
@Override
@Comment("重新加载配置")
public void reloadConfig(short paytype) {
if (this.conf != null && !this.conf.isEmpty()) { //存在微信支付配置
try {
File file = (this.conf.indexOf('/') == 0 || this.conf.indexOf(':') > 0) ? new File(this.conf) : new File(home, "conf/" + this.conf);
InputStream in = (file.isFile() && file.canRead()) ? new FileInputStream(file) : getClass().getResourceAsStream("/META-INF/" + this.conf);
if (in == null) return;
Properties properties = new Properties();
properties.load(in);
in.close();
this.elements = WeixinPayElement.create(logger, properties, home);
} catch (Exception e) {
logger.log(Level.SEVERE, "init weixinpay conf error", e);
}
}
}
public void setPayElements(Map elements) {
this.elements = elements;
}
public void putPayElements(Map elements) {
this.elements.putAll(elements);
}
@Override
public WeixinPayElement getPayElement(String appid) {
return this.elements.get(appid);
}
public void setPayElement(String appid, WeixinPayElement element) {
this.elements.put(appid, element);
}
public boolean existsPayElement(String appid) {
return this.elements != null && this.elements.containsKey(appid);
}
/**
* 手机支付或者微信公众号支付时调用
*
* @param request PayPreRequest
*
* @return PayPreResponse
*/
@Override
public PayPreResponse prepay(PayPreRequest request) {
return prepayAsync(request).join();
}
@Override
public CompletableFuture prepayAsync(PayPreRequest request) {
request.checkVaild();
final PayPreResponse result = new PayPreResponse();
try {
final WeixinPayElement element = elements.get(request.getAppid());
if (element == null) return result.retcode(RETPAY_CONF_ERROR).toFuture();
result.setAppid(element.appid);
final TreeMap map = new TreeMap<>();
if (request.getAttach() != null) map.putAll(request.getAttach());
map.put("appid", element.appid);
map.put("mch_id", element.merchno);
map.put("nonce_str", Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime()));
map.put("body", request.getPaybody());
//map.put("attach", "" + payid);
map.put("out_trade_no", request.getPayno());
map.put("total_fee", "" + request.getPaymoney());
map.put("spbill_create_ip", request.getClientAddr());
map.put("time_expire", String.format(format, System.currentTimeMillis() + request.getTimeoutms() * 60 * 1000));
map.put("notify_url", ((request.notifyurl != null && !request.notifyurl.isEmpty()) ? request.notifyurl : element.notifyurl));
map.put("trade_type", request.getPayway() == PAYWAY_WEB ? "JSAPI" : (request.getPayway() == PAYWAY_H5 ? "MWEB" : "APP"));
//trade_type=JSAPI,openid参数必传,用户在商户appid下的唯一标识
if (request.getPayway() == PAYWAY_WEB && !map.containsKey("openid")) return result.retcode(RETPAY_OPENID_ERROR).toFuture();
map.put("sign", createSign(element, map));
return postHttpContentAsync("https://api.mch.weixin.qq.com/pay/unifiedorder", formatMapToXML(map)).thenApply(responseText -> {
result.setResponsetext(responseText);
Map resultmap = formatXMLToMap(responseText);
if (!"SUCCESS".equals(resultmap.get("return_code"))) return result.retcode(RETPAY_PAY_ERROR);
if (!checkSign(element, resultmap)) return result.retcode(RETPAY_FALSIFY_ERROR);
/**
* "appId" : "wx2421b1c4370ec43b", //公众号名称,由商户传入 "timeStamp":" 1395712654", //时间戳,自1970年以来的秒数 "nonceStr" : "e61463f8efa94090b1f366cccfbbb444", //随机串 "package" :
* "prepay_id=u802345jgfjsdfgsdg888", "signType" : "MD5", //微信签名方式: "paySign" : "70EA570631E4BB79628FBCA90534C63FF7FADD89" //微信签名
*/
final String timestamp = Long.toString(System.currentTimeMillis() / 1000);
final String noncestr = Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime());
final Map retmap = new TreeMap<>();
if (request.getPayway() == PAYWAY_WEB) {
retmap.put("appId", element.appid);
retmap.put("timeStamp", timestamp);
retmap.put("nonceStr", noncestr);
retmap.put("package", "prepay_id=" + resultmap.get("prepay_id"));
retmap.put("signType", "MD5");
retmap.put("paySign", createSign(element, retmap));
} else {
retmap.put("appid", element.appid);
retmap.put("partnerid", element.merchno);
retmap.put("prepayid", resultmap.get("prepay_id"));
retmap.put("timestamp", timestamp);
retmap.put("noncestr", noncestr);
retmap.put("package", "Sign=WXPay"); //固定值
retmap.put("sign", createSign(element, retmap));
}
result.setResult(retmap);
return result;
});
} catch (Exception e) {
result.setRetcode(RETPAY_PAY_ERROR);
logger.log(Level.WARNING, "prepay_pay_error req=" + request + ", resp=" + result.responsetext, e);
return result.toFuture();
}
}
/**
* <xml>
* <appid><![CDATA[wx2421b1c4370ec43b]]></appid>
* <attach><![CDATA[支付测试]]></attach>
* <bank_type><![CDATA[CFT]]></bank_type>
* <fee_type><![CDATA[CNY]]></fee_type>
* <is_subscribe><![CDATA[Y]]></is_subscribe>
* <mch_id><![CDATA[10000100]]></mch_id>
* <nonce_str><![CDATA[5d2b6c2a8db53831f7eda20af46e531c]]></nonce_str>
* <openid><![CDATA[oUpF8uMEb4qRXf22hE3X68TekukE]]></openid>
* <out_trade_no><![CDATA[1409811653]]></out_trade_no>
* <result_code><![CDATA[SUCCESS]]></result_code>
* <return_code><![CDATA[SUCCESS]]></return_code>
* <sign><![CDATA[B552ED6B279343CB493C5DD0D78AB241]]></sign>
* <sub_mch_id><![CDATA[10000100]]></sub_mch_id>
* <time_end><![CDATA[20140903131540]]></time_end>
* <total_fee>1</total_fee>
* <trade_type><![CDATA[JSAPI]]></trade_type>
* <transaction_id><![CDATA[1004400740201409030005092168]]></transaction_id>
* </xml>
*
* @param request PayNotifyRequest
*
* @return PayNotifyResponse
*/
@Override
public PayNotifyResponse notify(PayNotifyRequest request) {
return notifyAsync(request).join();
}
@Override
public CompletableFuture notifyAsync(PayNotifyRequest request) {
request.checkVaild();
final PayNotifyResponse result = new PayNotifyResponse();
result.setPaytype(request.getPaytype());
final String rstext = " ";
Map map = formatXMLToMap(request.getText());
String appid = request.getAppid();
if (appid == null || appid.isEmpty()) appid = map.getOrDefault("appid", "");
final WeixinPayElement element = elements.get(appid);
if (element == null) return result.retcode(RETPAY_CONF_ERROR).toFuture();
result.setPayno(map.getOrDefault("out_trade_no", ""));
result.setThirdpayno(map.getOrDefault("transaction_id", ""));
if ("NOTPAY".equals(map.get("return_code"))) return result.retcode(RETPAY_PAY_WAITING).notifytext(rstext).toFuture();
if (!"SUCCESS".equals(map.get("return_code"))) return result.retcode(RETPAY_PAY_FAILED).notifytext(rstext).toFuture();
if (!(map instanceof SortedMap)) map = new TreeMap<>(map);
if (!checkSign(element, map)) return result.retcode(RETPAY_FALSIFY_ERROR).notifytext(rstext).toFuture();
String state = map.get("trade_state");
if (state == null && "SUCCESS".equals(map.get("result_code")) && Long.parseLong(map.get("total_fee")) > 0) {
state = "SUCCESS";
result.setPayedmoney(Long.parseLong(map.get("total_fee")));
}
if (!"SUCCESS".equals(state)) return result.retcode(RETPAY_PAY_FAILED).notifytext(rstext).toFuture();
return result.notifytext(rstext).toFuture();
}
/**
* 网页支付
*
* @param request PayCreatRequest
*
* @return PayCreatResponse
*/
@Override
public PayCreatResponse create(PayCreatRequest request) {
return createAsync(request).join();
}
@Override
public CompletableFuture createAsync(PayCreatRequest request) {
request.checkVaild();
final PayCreatResponse result = new PayCreatResponse();
try {
final WeixinPayElement element = elements.get(request.getAppid());
if (element == null) return result.retcode(RETPAY_CONF_ERROR).toFuture();
final TreeMap map = new TreeMap<>();
if (request.getAttach() != null) map.putAll(request.getAttach());
map.put("appid", element.appid);
map.put("mch_id", element.merchno);
map.put("nonce_str", Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime()));
map.put("body", request.getPaybody());
//map.put("attach", "" + payid);
map.put("out_trade_no", request.getPayno());
map.put("total_fee", "" + request.getPaymoney());
map.put("spbill_create_ip", request.getClientAddr());
map.put("time_expire", String.format(format, System.currentTimeMillis() + request.getPaytimeout() * 1000));
map.put("notify_url", element.notifyurl);
map.put("sign", createSign(element, map));
return postHttpContentAsync("https://api.mch.weixin.qq.com/pay/unifiedorder", formatMapToXML(map)).thenApply(responseText -> {
result.setResponsetext(responseText);
Map resultmap = formatXMLToMap(responseText);
if (!"SUCCESS".equals(resultmap.get("return_code"))) return result.retcode(RETPAY_PAY_ERROR);
if (!checkSign(element, resultmap)) return result.retcode(RETPAY_FALSIFY_ERROR);
/**
* "appId" : "wx2421b1c4370ec43b", //公众号名称,由商户传入 "timeStamp":" 1395712654", //时间戳,自1970年以来的秒数 "nonceStr" : "e61463f8efa94090b1f366cccfbbb444", //随机串 "package" :
* "prepay_id=u802345jgfjsdfgsdg888", "signType" : "MD5", //微信签名方式: "paySign" : "70EA570631E4BB79628FBCA90534C63FF7FADD89" //微信签名
*/
final Map rmap = new TreeMap<>();
result.setResult(rmap);
rmap.put("appId", element.appid);
rmap.put("timeStamp", Long.toString(System.currentTimeMillis() / 1000));
rmap.put("nonceStr", Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime()));
rmap.put("package", "prepay_id=" + resultmap.get("prepay_id"));
rmap.put("signType", "MD5");
rmap.put("paySign", createSign(element, rmap));
return result;
});
} catch (Exception e) {
result.setRetcode(RETPAY_PAY_ERROR);
logger.log(Level.WARNING, "create_pay_error req=" + request + ", resp=" + result.responsetext, e);
return result.toFuture();
}
}
/**
* 回调支付接口返回的结果
* < p >
* < xml >
* < appid >< ![CDATA[wx4ad12c89818dd981]] >< /appid >
* < attach >< ![CDATA[10000070334]] >< /attach >
* < bank_type >< ![CDATA[ICBC_DEBIT]] >< /bank_type >
* < cash_fee >< ![CDATA[10]] >< /cash_fee >
* < fee_type >< ![CDATA[CNY]] >< /fee_type >
* < is_subscribe >< ![CDATA[Y]] >< /is_subscribe >
* < mch_id >< ![CDATA[1241384602]] >< /mch_id >
* < nonce_str >< ![CDATA[14d69ac6d6525f27dc9bcbebc]] >< /nonce_str >
* < openid >< ![CDATA[ojEVbsyDUzGqlgX3eDgmAMaUDucA]] >< /openid >
* < out_trade_no >< ![CDATA[1000072334]] >< /out_trade_no >
* < result_code >< ![CDATA[SUCCESS]] >< /result_code >
* < return_code >< ![CDATA[SUCCESS]] >< /return_code >
* < sign >< ![CDATA[60D95E25EA9C4F54BD1020952303C4E2]] >< /sign >
* < time_end >< ![CDATA[20150519085546]] >< /time_end >
* < total_fee >10< /total_fee >
* < trade_type >< ![CDATA[JSAPI]] >< /trade_type >
* < transaction_id >< ![CDATA[1009630061201505190139511926]] >< /transaction_id >
* < /xml >
*
* @param request PayRequest
*
* @return PayQueryResponse
*/
@Override
public PayQueryResponse query(PayRequest request) {
return queryAsync(request).join();
}
@Override
public CompletableFuture queryAsync(PayRequest request) {
request.checkVaild();
final PayQueryResponse result = new PayQueryResponse();
try {
final WeixinPayElement element = elements.get(request.getAppid());
if (element == null) return result.retcode(RETPAY_CONF_ERROR).toFuture();
final Map map = new TreeMap<>();
map.put("appid", element.appid);
map.put("mch_id", element.merchno);
map.put("out_trade_no", request.getPayno());
map.put("nonce_str", Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime()));
map.put("sign", createSign(element, map));
return postHttpContentAsync("https://api.mch.weixin.qq.com/pay/orderquery", formatMapToXML(map)).thenApply(responseText -> {
result.setResponsetext(responseText);
final Map resultmap = formatXMLToMap(responseText);
result.setResult(resultmap);
String state = resultmap.getOrDefault("trade_state", "");
if (state.isEmpty() && "SUCCESS".equals(resultmap.get("result_code")) && Long.parseLong(resultmap.get("total_fee")) > 0) {
state = "SUCCESS";
}
if (!checkSign(element, resultmap)) return result.retcode(RETPAY_FALSIFY_ERROR);
if (state.isEmpty()) logger.warning("weixin.pay.query = " + resultmap);
//trade_state 支付状态: SUCCESS—支付成功 REFUND—转入退款 NOTPAY—未支付 CLOSED—已关闭 REVOKED—已撤销(刷卡支付) USERPAYING--用户支付中 PAYERROR--支付失败(其他原因,如银行返回失败)
short paystatus = PAYSTATUS_PAYNO;
switch (state) {
case "SUCCESS": paystatus = PAYSTATUS_PAYOK;
break;
case "NOTPAY": paystatus = PAYSTATUS_UNPAY;
break;
case "CLOSED": paystatus = PAYSTATUS_CLOSED;
break;
case "REVOKED": paystatus = PAYSTATUS_CANCELED;
break;
case "USERPAYING": paystatus = PAYSTATUS_PAYING;
break;
case "PAYERROR": paystatus = PAYSTATUS_PAYNO;
break;
}
result.setPaystatus(paystatus);
result.setThirdpayno(resultmap.getOrDefault("transaction_id", ""));
result.setPayedmoney(Long.parseLong(resultmap.getOrDefault("total_fee", "0")));
return result;
});
} catch (Exception e) {
result.setRetcode(RETPAY_PAY_ERROR);
logger.log(Level.WARNING, "query_pay_error req=" + request + ", resp=" + result.responsetext, e);
return result.toFuture();
}
}
@Override
public PayResponse close(PayCloseRequest request) {
return closeAsync(request).join();
}
@Override
public CompletableFuture closeAsync(PayCloseRequest request) {
request.checkVaild();
final PayResponse result = new PayResponse();
try {
final WeixinPayElement element = elements.get(request.getAppid());
if (element == null) return result.retcode(RETPAY_CONF_ERROR).toFuture();
Map map = new TreeMap<>();
map.put("appid", element.appid);
map.put("mch_id", element.merchno);
map.put("nonce_str", Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime()));
map.put("out_trade_no", request.getPayno());
map.put("sign", createSign(element, map));
return postHttpContentAsync("https://api.mch.weixin.qq.com/pay/closeorder", formatMapToXML(map)).thenApply(responseText -> {
result.setResponsetext(responseText);
Map resultmap = formatXMLToMap(responseText);
if (!"SUCCESS".equals(resultmap.get("return_code"))) return result.retcode(RETPAY_PAY_ERROR);
if (!checkSign(element, resultmap)) return result.retcode(RETPAY_FALSIFY_ERROR);
result.setResult(resultmap);
return result;
});
} catch (Exception e) {
result.setRetcode(RETPAY_PAY_ERROR);
logger.log(Level.WARNING, "close_pay_error req=" + request + ", resp=" + result.responsetext, e);
return result.toFuture();
}
}
@Override
public PayRefundResponse refund(PayRefundRequest request) {
return refundAsync(request).join();
}
//https://pay.weixin.qq.com/wiki/doc/api/micropay.php?chapter=9_4
@Override
public CompletableFuture refundAsync(PayRefundRequest request) {
request.checkVaild();
final PayRefundResponse result = new PayRefundResponse();
try {
final WeixinPayElement element = elements.get(request.getAppid());
if (element == null) return result.retcode(RETPAY_CONF_ERROR).toFuture();
final TreeMap map = new TreeMap<>();
map.put("appid", element.appid);
map.put("mch_id", element.merchno);
map.put("nonce_str", Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime()));
map.put("out_trade_no", request.getPayno());
map.put("total_fee", "" + request.getPaymoney());
map.put("out_refund_no", request.getRefundno());
map.put("refund_fee", "" + request.getRefundmoney());
map.put("op_user_id", element.merchno);
map.put("sign", createSign(element, map));
return postHttpContentAsync(element.paySSLContext, "https://api.mch.weixin.qq.com/secapi/pay/refund", formatMapToXML(map)).thenApply(responseText -> {
result.setResponsetext(responseText);
Map resultmap = formatXMLToMap(responseText);
if (!"SUCCESS".equals(resultmap.get("return_code"))) return result.retcode(RETPAY_REFUND_ERROR);
if (!checkSign(element, resultmap)) return result.retcode(RETPAY_FALSIFY_ERROR);
result.setRefundedmoney(Long.parseLong(resultmap.get("refund_fee")));
return result;
});
} catch (Exception e) {
result.setRetcode(RETPAY_REFUND_ERROR);
logger.log(Level.WARNING, "refund_pay_error req=" + request + ", resp=" + result.responsetext, e);
return result.toFuture();
}
}
@Override
public PayRefundResponse queryRefund(PayRequest request) {
return queryRefundAsync(request).join();
}
@Override
public CompletableFuture queryRefundAsync(PayRequest request) {
request.checkVaild();
final PayRefundResponse result = new PayRefundResponse();
try {
final WeixinPayElement element = elements.get(request.getAppid());
if (element == null) return result.retcode(RETPAY_CONF_ERROR).toFuture();
final Map map = new TreeMap<>();
map.put("appid", element.appid);
map.put("mch_id", element.merchno);
map.put("out_trade_no", request.getPayno());
map.put("nonce_str", Long.toHexString(System.currentTimeMillis()) + Long.toHexString(System.nanoTime()));
map.put("sign", createSign(element, map));
return postHttpContentAsync("https://api.mch.weixin.qq.com/pay/refundquery", formatMapToXML(map)).thenApply(responseText -> {
result.setResponsetext(responseText);
final Map resultmap = formatXMLToMap(responseText);
result.setResult(resultmap);
if (!"SUCCESS".equals(resultmap.get("return_code"))) return result.retcode(RETPAY_PAY_ERROR);
if (!checkSign(element, resultmap)) return result.retcode(RETPAY_FALSIFY_ERROR);
//trade_state SUCCESS—退款成功 FAIL—退款失败 PROCESSING—退款处理中 NOTSURE—未确定,需要商户原退款单号重新发起
//CHANGE—转入代发,退款到银行发现用户的卡作废或者冻结了,导致原路退款银行卡失败,资金回流到商户的现金帐号,需要商户人工干预,通过线下或者财付通转账的方式进行退款。
result.setResult(resultmap);
result.setRefundedmoney(Long.parseLong(resultmap.get("refund_fee_$n")));
return result;
});
} catch (Exception e) {
result.setRetcode(RETPAY_PAY_ERROR);
logger.log(Level.WARNING, "query_pay_error req=" + request + ", resp=" + result.responsetext, e);
return result.toFuture();
}
}
@Override
protected String createSign(final PayElement element, Map map) { //计算签名
final StringBuilder sb = new StringBuilder();
map.forEach((x, y) -> {
if (!((String) y).isEmpty()) sb.append(x).append('=').append(y).append('&');
});
sb.append("key=").append(((WeixinPayElement) element).signkey);
try {
return Utility.binToHexString(MessageDigest.getInstance("MD5").digest(sb.toString().getBytes())).toUpperCase();
} catch (Exception ex) {
throw new RuntimeException(ex);
}
}
@Override
protected boolean checkSign(final PayElement element, Map map) { //验证签名
if (!(map instanceof SortedMap)) map = new TreeMap<>(map);
String sign = (String) map.remove("sign");
final StringBuilder sb = new StringBuilder();
map.forEach((x, y) -> {
if (!((String) y).isEmpty()) sb.append(x).append('=').append(y).append('&');
});
sb.append("key=").append(((WeixinPayElement) element).signkey);
try {
return sign.equals(Utility.binToHexString(MessageDigest.getInstance("MD5").digest(sb.toString().getBytes())).toUpperCase());
} catch (Exception e) {
return false;
}
}
protected static String formatMapToXML(final Map map) {
final StringBuilder sb = new StringBuilder();
sb.append("");
map.forEach((x, y) -> sb.append('<').append(x).append('>').append(y.replace("<", "<").replace(">", ">").replace("&", "&")).append("").append(x).append('>'));
sb.append(" ");
return sb.toString();
}
public static Map formatXMLToMap(final String xml) {
Map map = new TreeMap<>();
Matcher m = PAYXML.matcher(xml.substring(xml.indexOf('>') + 1));
while (m.find()) {
String val = m.group(2);
if (val.startsWith(" create(Logger logger, Properties properties, File home) {
String def_appid = properties.getProperty("pay.weixin.appid", "").trim();
String def_merchno = properties.getProperty("pay.weixin.merchno", "").trim();
String def_submerchno = properties.getProperty("pay.weixin.submerchno", "").trim();
String def_notifyurl = properties.getProperty("pay.weixin.notifyurl", "").trim();
String def_signkey = properties.getProperty("pay.weixin.signkey", "").trim();
String def_certpwd = properties.getProperty("pay.weixin.certpwd", "").trim();
String def_certpath = properties.getProperty("pay.weixin.certpath", "").trim();
String def_certbase64 = properties.getProperty("pay.weixin.certbase64", "").trim();
final Map map = new HashMap<>();
properties.keySet().stream().filter(x -> x.toString().startsWith("pay.weixin.") && x.toString().endsWith(".appid")).forEach(appid_key -> {
final String prefix = appid_key.toString().substring(0, appid_key.toString().length() - ".appid".length());
String appid = properties.getProperty(prefix + ".appid", def_appid).trim();
String merchno = properties.getProperty(prefix + ".merchno", def_merchno).trim();
String submerchno = properties.getProperty(prefix + ".submerchno", def_submerchno).trim();
String notifyurl = properties.getProperty(prefix + ".notifyurl", def_notifyurl).trim();
String signkey = properties.getProperty(prefix + ".signkey", def_signkey).trim();
String certpwd = properties.getProperty(prefix + ".certpwd", def_certpwd).trim();
String certpath = properties.getProperty(prefix + ".certpath", def_certpath).trim();
String certbase64 = properties.getProperty(prefix + ".certbase64", def_certbase64).trim();
if (appid.isEmpty() || merchno.isEmpty() || notifyurl.isEmpty() || signkey.isEmpty()) {
logger.log(Level.WARNING, properties + "; has illegal weixinpay conf by prefix" + prefix);
return;
}
WeixinPayElement element = new WeixinPayElement();
element.appid = appid;
element.merchno = merchno;
element.submerchno = submerchno;
element.notifyurl = notifyurl;
element.signkey = signkey;
element.certpwd = certpwd;
element.certpath = certpath;
element.certbase64 = certbase64;
if (element.initElement(logger, home)) {
map.put(appid, element);
if (def_appid.equals(appid)) map.put("", element);
}
});
return map;
}
@Override
public boolean initElement(Logger logger, File home) {
if ((this.certbase64 == null || this.certbase64.isEmpty())
&& (this.certpath == null || this.certpath.isEmpty())) return true;
try {
InputStream in;
if (this.certbase64 != null && !this.certbase64.isEmpty()) {
in = new ByteArrayInputStream(Base64.getDecoder().decode(this.certbase64));
} else {
File file = (certpath.indexOf('/') == 0 || certpath.indexOf(':') > 0) ? new File(this.certpath) : new File(home, "conf/" + this.certpath);
in = file.isFile() ? new FileInputStream(file) : getClass().getResourceAsStream("/META-INF/" + this.certpath);
}
if (in == null) return false;
//需要更新%JDK_HOME%\jre\lib\security下的policy
//http://www.oracle.com/technetwork/java/javase/downloads/jce8-download-2133166.html
setPaySSLContext(in);
return true;
} catch (Exception e) {
logger.log(Level.SEVERE, "init weixinpay sslcontext error", e);
return false;
}
}
public void setPaySSLContext(InputStream in) throws Exception {
final KeyStore keyStore = KeyStore.getInstance("PKCS12");
keyStore.load(in, this.certpwd.toCharArray());
in.close();
KeyManagerFactory keyManagerFactory = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
keyManagerFactory.init(keyStore, certpwd.toCharArray());
SSLContext ctx = SSLContext.getInstance("TLSv1");
ctx.init(keyManagerFactory.getKeyManagers(), null, null);
this.paySSLContext = ctx;
}
@Override
public String toString() {
return JsonConvert.root().convertTo(this);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy