- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
在微信小程序中实现一个提现功能 ,提现的钱其实来自系统中的其他功能。
当然这个功能,需要一个微信支付的商户号,并且账户中需要有充足的余额。
注:API证书文件的后缀是.p12
<dependency>
<groupId>com.github.wxpay</groupId>
<artifactId>wxpay-sdk</artifactId>
<version>0.0.3</version>
</dependency>
withDraw:
apiKey: xxxxxxxx
mchid: xxxxxx
mch_appid: xxxxxxxxx
//此路径代表是在项目的resouce根目录
certPath: /xxxxxxxx.p12
count: 10
quota: 200
@Component
public class WithDrawConfig implements WXPayConfig {
//从yaml注入配置
@Value("${withDraw.mch_appid}")
private String mch_appid;
@Value("${withDraw.apiKey}")
private String apiKey;
@Value("${withDraw.mchid}")
private String mchid;
@Value("${withDraw.certPath}")
private String certPath;
@Override
public String getAppID() {
return mch_appid;
}
@Override
public String getMchID() {
return mchid;
}
@Override
public String getKey() {
return apiKey;
}
@Override
public InputStream getCertStream() {
//获取证书,证书建议放到resource目录下
return this.getClass().getResourceAsStream(certPath);
}
@Override
public int getHttpConnectTimeoutMs() {
return 8000;
}
@Override
public int getHttpReadTimeoutMs() {
return 10000;
}
}
@Slf4j
@Component
@RequiredArgsConstructor
public class WithDrawUtils {
@Value("${withDraw.mch_appid}")
private String mch_appid;
@Value("${withDraw.mchid}")
private String mchid;
@Value("${withDraw.apiKey}")
private String apiKey;
private final WithDrawConfig withDrawConfig;
//生成订单号日期时间+随机字符
public String getOrderNumber() {
SimpleDateFormat date = new SimpleDateFormat("yyyyMMddHHmmss");
return date.format(new Date()) + RandomStringUtils.randomNumeric(4);
}
public Map<String, String> fillRequest(Map<String, String> reqData) throws Exception {
reqData.put("mch_appid", mch_appid);
reqData.put("mchid", mchid);
reqData.put("nonce_str", WXPayUtil.generateNonceStr().toUpperCase());
reqData.put("sign", WXPayUtil.generateSignature(reqData, apiKey, WXPayConstants.SignType.MD5));
return reqData;
}
public String getMd5ByString(String str) {
StringBuilder hexString = new StringBuilder();
try {
MessageDigest mdTemp = MessageDigest.getInstance("MD5");
mdTemp.update(str.getBytes());
byte[] hash = mdTemp.digest();
for (byte b : hash) {
if ((0xff & b) < 0x10) {
hexString.append("0").append(Integer.toHexString((0xFF & b)));
} else {
hexString.append(Integer.toHexString(0xFF & b));
}
}
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
}
return hexString.toString();
}
/**
* 将对象直接转换成String类型的 XML输出
*
* @param obj
* @return
*/
public String convertToXml(Object obj) {
// 创建输出流
StringWriter sw = new StringWriter();
try {
// 利用jdk中自带的转换类实现
JAXBContext context = JAXBContext.newInstance(obj.getClass());
Marshaller marshaller = context.createMarshaller();
// 格式化xml输出的格式
marshaller.setProperty(Marshaller.JAXB_FRAGMENT,
Boolean.TRUE);
// 将对象转换成输出流形式的xml
marshaller.marshal(obj, sw);
} catch (JAXBException e) {
e.printStackTrace();
}
return sw.toString();
}
public String getRestInstance(String url, String data) throws Exception {
String UTF8 = "UTF-8";
URL httpUrl = new URL(url);
char[] password = mchid.toCharArray();//证书密码
InputStream certStream = withDrawConfig.getCertStream();//获取证书的流
KeyStore ks = KeyStore.getInstance("PKCS12");
ks.load(certStream, password);
// 实例化密钥库 & 初始化密钥工厂
KeyManagerFactory kmf = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm());
kmf.init(ks, password);
SSLContext sslContext = SSLContext.getInstance("TLS");
sslContext.init(kmf.getKeyManagers(), (TrustManager[]) null, new SecureRandom());
HttpsURLConnection.setDefaultSSLSocketFactory(sslContext.getSocketFactory());
HttpURLConnection httpURLConnection = (HttpURLConnection) httpUrl.openConnection();
httpURLConnection.setDoOutput(true);
httpURLConnection.setRequestMethod("POST");
httpURLConnection.setConnectTimeout(withDrawConfig.getHttpConnectTimeoutMs());
httpURLConnection.setReadTimeout(withDrawConfig.getHttpReadTimeoutMs());
httpURLConnection.connect();
OutputStream outputStream = httpURLConnection.getOutputStream();
outputStream.write(data.getBytes(UTF8));
InputStream inputStream = httpURLConnection.getInputStream();
BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream, UTF8));
StringBuilder stringBuffer = new StringBuilder();
String line;
while ((line = bufferedReader.readLine()) != null) {
stringBuffer.append(line);
}
String resp = stringBuffer.toString();
try {
bufferedReader.close();
} catch (IOException ignored) {
}
try {
inputStream.close();
} catch (IOException ignored) {
}
try {
outputStream.close();
} catch (IOException ignored) {
}
if (certStream != null) {
try {
certStream.close();
} catch (IOException ignored) {
}
}
return resp;
}
}
官方接口文档:【微信支付】付款开发者文档 (qq.com)
//开始提现,生成订单号
String orderNumber = withDrawUtils.getOrderNumber();
//自定义的将提现所需要的参数封装的实体类
WithDrawDTO withDrawDTO = new WithDrawDTO();
withDrawDTO.setPartner_trade_no(orderNumber);
withDrawDTO.setDesc("xxxxx");
withDrawDTO.setAmount(appletWithDrawDTO.getAmount());
//此参数代表,开启真实姓名校验,也可以关闭,详看官方文档的参数说明
withDrawDTO.setCheck_name("FORCE_CHECK");
withDrawDTO.setRe_user_name(appletWithDrawDTO.getName());
//微信小程序用户的openid
withDrawDTO.setOpenid(wxUser.getOpenid());
Map<String, String> params = JSON.parseObject(JSON.toJSONString(withDrawDTO), new TypeReference<Map<String, String>>() {
});
params = withDrawUtils.fillRequest(params);
withDrawDTO.setNonce_str(params.get("nonce_str"));
withDrawDTO.setMchid(params.get("mchid"));
withDrawDTO.setMch_appid(params.get("mch_appid"));
withDrawDTO.setSign(params.get("sign"));
String url = "https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers";
String post = withDrawUtils.getRestInstance(url, withDrawUtils.convertToXml(withDrawDTO));
Map<String, String> result = WXPayUtil.xmlToMap(post);
//result为调用接口之后的返回参数,可以根据返回参数判断是否成功
WithDrawEnum resultEnum;
if ("SUCCESS".equals(result.get("result_code"))) {
//提现成功
resultEnum = WithDrawEnum.fromText("SUCCESS");
return resultEnum;
} else {
//提现失败
resultEnum = WithDrawEnum.fromText(result.get("err_code"));
throw new BadRequestException(resultEnum);
}
@Getter
@NoArgsConstructor
@AllArgsConstructor
public enum WithDrawEnum {
SUCCESS(0, "提现成功"),
FAIL(400, "提现失败,余额小于提现金额"),
HANDLE_FRE(5000, "操作太频繁"),
NO_AUTH(5001, "没有该接口权限"),
AMOUNT_LIMIT(5002, "金额超限"),
PARAM_ERROR(5003, "参数错误"),
OPENID_ERROR(5004, "Openid错误"),
SEND_FAILED(5005, "付款错误"),
NOTENOUGH(5006, "余额不足"),
SYSTEMERROR(5007, "系统繁忙,请稍后再试。"),
NAME_MISMATCH(5008, "姓名校验出错"),
SIGN_ERROR(5009, "签名错误"),
XML_ERROR(5010, "发送内容出错"),
FATAL_ERROR(5011, "两次发送参数不一致"),
FREQ_LIMIT(5012, "超过频率限制,请稍后再试。"),
MONEY_LIMIT(5013, "已经达到今日付款总额上限/已达到付款给此用户额度上限"),
CA_ERROR(5014, "商户证书校验出错"),
V2_ACCOUNT_SIMPLE_BAN(5015, "无法给未实名用户付款"),
PARAM_IS_NOT_UTF8(5016, "发送参数中包含不规范字符"),
SENDNUM_LIMIT(5017, "该用户今日付款次数超过限制, 如有需要请进入【微信支付商户平台-产品中心-付款到零钱-产品设置】进行修改"),
RECV_ACCOUNT_NOT_ALLOWED(5018, "收款账户不在收款账户列表"),
PAY_CHANNEL_NOT_ALLOWED(5019, "本商户号未配置此功能"),
SEND_MONEY_LIMIT(5020, "已达到今日商户付款额度上限"),
RECEIVED_MONEY_LIMIT(5021, "已达到今日付款给此用户额度上限");
private int code;
private String msg;
public static WithDrawEnum fromText(String text) {
if (text != null) {
for (WithDrawEnum b : WithDrawEnum.values()) {
if (text.equalsIgnoreCase(b.name())) {
return b;
}
}
}
return null;
}
}
微信相关接口的调用,主要还是要耐心看文档的说明,才能实现业务需求。
我有一个包含交易的简单表格,我想了解每个月有多少消费者进行的交易总额大于 0,并且他们的第一笔交易不在该月。首次交易是指客户在当月首次购买。 我试图得到的结果是以下形式: +--------+----
这是一个网站,点击“Dodaję”(在波兰语中意为“添加”)后,商品进入购物篮 30 分钟:https://remix.pl/ 当元素在购物篮中时,其他人无法购买该元素。我检查了他们的 basked:
我想知道,您是否编写了一个使用 Paypal 付款作为付款方式的网站(您被重定向到他们的网站进行付款的付款方式,而不是网络付款专业版)。 我可以存储什么样的详细信息?是否只有他们的 Paypal 电子
我一直在开发需要实现触碰付款的应用程序。我能够将 HCE 服务与 NFC 终端连接。 现在我的问题是接下来的步骤是什么,用它进行实际付款? 我到处搜索,但找不到合适的文件。请帮我。 下面是我编写的将
我正在为我在学校的最终项目开发一个网站,在我的网站上您可以购买硬币来打开礼物,但我想自动付款。当客户完成付款后,他会被重定向到 success.php,我如何安排他在付款成功后自动在他的帐户中获得 X
我正在使用 Stripe 在我的应用程序中付款。我成功获得了 token 。我想知道如何收费,因为我的 Stripe 仪表板中没有反射(reflect)任何变化。我正在使用我的 Stripe 帐户 T
我的网站是用 Python 编码的,我想提供通过 paypal 在我的网站上接受付款的功能。我想使用免费的 paypal 服务(无月费)来执行此操作,用户将被重定向到 paypals 站点以登录并付款
我们正在开发一个 iOS 应用程序,它提供一些可以在应用程序内使用的积分(付费)。我们对此几乎没有疑问 - 我们能否使用第三方支付网关,如 paypal 或类似的网关,或者我们需要为此实现 IAP?
我想从我的账户处理向 PayPal 用户的 PayPal 付款。我使用 PHP 脚本从我的服务器获取表单详细信息,但是,我无法仅使用 PHP 脚本处理付款(我想 PayPal 网站要求用户在客户端使用
我正在尝试实现以下场景: 用户批准针对选定包大小的定期每周付款(订阅) 用户每周都可以:A。取消特定周的交货并避免支付本周的费用b.修改包裹尺寸,只为这次少付/多付C。除了每周套餐之外,添加其他项目并
我尝试用信用卡创建付款并从 paypal 得到回复成功 ACK 但在 IPN 中我的目标是获得和等待响应,为什么我在尝试使用此付款时收到此错误 METHOD=DoDirectPayment &VERS
您能否以某种方式要求用户批准付款并在稍后处理这笔付款?具体来说,我有一个市场,用户 A 可以在其中向其他用户 B 索取一些产品。我想在用户 A 收到他的产品时收费,但我希望用户 B 确保不会白白创造他
我正在创建一个 phonegap 应用程序,它有 2 个变量用于在 paypal.js 文件中设置 PayPalEnvironmentProduction 和 PayPalEnvironmentSan
我的问题可能看起来非常基础并且依赖于 View ,但我希望了解它的各个方面。我创建了一个移动应用程序,用户需要在其中注册才能使用该应用程序,但我的客户希望拥有网站形式的管理面板,用户可以通过 payp
已关闭。此问题旨在寻求有关书籍、工具、软件库等的建议。不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,
我在我的网站上付款,您可以在网站上赚钱。因此,用户可以选择他想向他的网站账户转账多少钱。 我希望可以选择当用户的资金少于例如 20 美元时自动启动从他的帐户转账到我的帐户并在我的网站上充值他的信用额度
我想创建一个市场,用户可以在其中列出商品并让买家直接通过 Paypal 付款。有没有办法在我们这边跟踪这些付款,以便我可以对数据进行操作(例如将订单标记为已完成)?我在想我可以让卖家将他们的 payp
我目前在我工作的网站上使用 Paypal 作为支付系统,到目前为止一切正常...除了一件事。付款完成后,Paypal 调用我提供的 IPN 页面,一切正常。但是,如果此人取消付款,我不确定我应该如何知
我有两个网站,两个网站都通过 PayPal 付款销售相同的产品。但现在我需要确定特定交易是从哪个网站进行的。我的意思是这次购买是从哪个网站进行的。那么有什么方法可以识别吗?? 非常感谢。 最佳答案 您
我想在paypal支付页面获取多个产品。我现在只得到一个单一的第一个产品。有人可以帮我解决我错的地方吗?下面是我的代码..我搜索谷歌,发现还有两件事要改变。即 _xclick 到 _cart 和一个
我是一名优秀的程序员,十分优秀!