​苹果内购:

只要你在苹果系统购买APP中虚拟物品(虚拟货币,VIP充值等),必须通过内购方式进行支付,苹果和商家进行三七开

验证模式有两种:

Validating Receipts With the App Store 通过访问苹果接口进行验证。
Validating Receipts Locally 本地代码解码进行验证

官方验证文档地址:https://developer.apple.com/library/archive/releasenotes/General/ValidateAppStoreReceipt/Chapters/ValidateRemotely.html#//apple_ref/doc/uid/TP40010573-CH104-SW1

官方文档说明

我这里主要说一下服务端验证模式,大致流程为

  • app进行支付,然后收到苹果的收据(一串很长的BASE64编码的字符串)
  • app请求服务端,将收据给到服务端,服务端拿到收据请求苹果服务器验证收据是否为真
  • 服务端验证收据真伪,验证当前支付的交易是否成功,成功则处理支付成功的业务逻辑

进行代码前,首先使用postman将收据发送给苹果服务器,熟悉一下返回的数据结构

重点说一下我的理解

在官方文档和各个私人博客中都没有明确说明要验证的内容,百度一整天得到的验证逻辑为
苹果服务器只验证了收据的真伪,而收据包含多个交易的信息。
所以,我们验证当status字段为0(即收据为真),且当前交易ID(app传递到后台)在收据交易列表中,即可认为交易支付成功
同时app传递当前支付产品的ID(我们内部的商品ID),处理该商品的订单

注意:这个接口可以多次请求,所以应当将交易ID与订单进行绑定,防止一个交易生成多个订单

上验证代码,首先来一个百度的工具类,功能为组装请求数据,发送http请求

import javax.net.ssl.*;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.URL;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
import java.util.Locale;
​
/*** 苹果IAP内购验证工具类* Created by wangqichang on 2019/2/26.*/
public class IosVerifyUtil {private static class TrustAnyTrustManager implements X509TrustManager {
​public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
​public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
​public X509Certificate[] getAcceptedIssuers() {return new X509Certificate[] {};}}
​private static class TrustAnyHostnameVerifier implements HostnameVerifier {public boolean verify(String hostname, SSLSession session) {return true;}}
​private static final String url_sandbox = "https://sandbox.itunes.apple.com/verifyReceipt";private static final String url_verify = "https://buy.itunes.apple.com/verifyReceipt";
​/*** 苹果服务器验证** @param receipt*            账单* @url 要验证的地址* @return null 或返回结果 沙盒 https://sandbox.itunes.apple.com/verifyReceipt**/public static String buyAppVerify(String receipt,int type) {//环境判断 线上/开发环境用不同的请求链接String url = "";if(type==0){url = url_sandbox; //沙盒测试}else{url = url_verify; //线上测试}//String url = EnvUtils.isOnline() ?url_verify : url_sandbox;
​try {SSLContext sc = SSLContext.getInstance("SSL");sc.init(null, new TrustManager[] { new TrustAnyTrustManager() }, new java.security.SecureRandom());URL console = new URL(url);HttpsURLConnection conn = (HttpsURLConnection) console.openConnection();conn.setSSLSocketFactory(sc.getSocketFactory());conn.setHostnameVerifier(new TrustAnyHostnameVerifier());conn.setRequestMethod("POST");conn.setRequestProperty("content-type", "text/json");conn.setRequestProperty("Proxy-Connection", "Keep-Alive");conn.setDoInput(true);conn.setDoOutput(true);BufferedOutputStream hurlBufOus = new BufferedOutputStream(conn.getOutputStream());
​String str = String.format(Locale.CHINA, "{"receipt-data":"" + receipt + ""}");//拼成固定的格式传给平台hurlBufOus.write(str.getBytes());hurlBufOus.flush();
​InputStream is = conn.getInputStream();BufferedReader reader = new BufferedReader(new InputStreamReader(is));String line = null;StringBuffer sb = new StringBuffer();while ((line = reader.readLine()) != null) {sb.append(line);}
​return sb.toString();} catch (Exception ex) {System.out.println("苹果服务器异常");ex.printStackTrace();}return null;}
​/*** 用BASE64加密** @param str* @return*/public static String getBASE64(String str) {byte[] b = str.getBytes();String s = null;if (b != null) {s = new sun.misc.BASE64Encoder().encode(b);}return s;}
}

验证逻辑代码

/*** 苹果内购校验* @param priceId 会员价格ID* @param transactionId 苹果内购交易ID* @param payload 校验体(base64字符串)* @return*/@PostMapping("/iospay")public Map<String, Object> iosPay(Long priceId,String transactionId, String payload) {log.info("苹果内购校验开始,交易ID:" + transactionId + " base64校验体:" + payload);Shipper shipper = getLoginShipper();if (shipper == null) {return failure("未登录");}
​//线上环境验证String verifyResult = IosVerifyUtil.buyAppVerify(payload, 1);if (verifyResult == null) {return failure("苹果验证失败,返回数据为空");} else {log.info("线上,苹果平台返回JSON:" + verifyResult);JSONObject appleReturn = JSONObject.parseObject(verifyResult);String states = appleReturn.getString("status");//无数据则沙箱环境验证if ("21007".equals(states)) {verifyResult = IosVerifyUtil.buyAppVerify(payload, 0);log.info("沙盒环境,苹果平台返回JSON:" + verifyResult);appleReturn = JSONObject.parseObject(verifyResult);states = appleReturn.getString("status");}log.info("苹果平台返回值:appleReturn" + appleReturn);// 前端所提供的收据是有效的    验证成功if (states.equals("0")) {String receipt = appleReturn.getString("receipt");JSONObject returnJson = JSONObject.parseObject(receipt);String inApp = returnJson.getString("in_app");List<HashMap> inApps = JSONObject.parseArray(inApp, HashMap.class);if (!CollectionUtils.isEmpty(inApps)) {ArrayList<String> transactionIds = new ArrayList<String>();for (HashMap app : inApps) {transactionIds.add((String) app.get("transaction_id"));}//交易列表包含当前交易,则认为交易成功if (transactionIds.contains(transactionId)) {//处理业务逻辑VipOrder vipOrder = vipOrderService.saveVipOrder(shipper, priceId, EnumPayType.APPLE_IN_APP_PURCHASES.getValue(),transactionId);vipOrderService.paySuccess(vipOrder.getOrderCode(),null);log.info("交易成功,新增并处理订单:{}",vipOrder.getOrderCode());return success("充值成功");}return failure("当前交易不在交易列表中");}return failure("未能获取获取到交易列表");} else {return failure("支付失败,错误码:" + states);}}}

同学们可以直接copy,注意删除我个人的业务代码即可

这个时候ios开发提出了一个问题,当支付完成,还没有发起验证,app闪退或关机时,岂不是无法生成订单?

这个时候可以先保存到本地,每次app启动判断本地是否有尚未验证的交易,有则发起验证请求。验证返回成功则删除本地记录

postman关闭ssl验证_【第5期】springboot:苹果内购服务端验证相关推荐

  1. ios内购php验证码,PHP (Laravel) 实现 iOS 内购服务端验证

    /** * @Author woann * @param Request $request * @return \Illuminate\Http\JsonResponse * @des ios内购支付 ...

  2. Python Google内购服务端验证

    Google内购完成后,服务端需要校验订单的状态是否正确(是否已经成功付款). 一.申请认证 参考https://developers.google.cn/android-publisher/gett ...

  3. golang 苹果登录,服务端验证identityToken(真实有效)

    介绍 2019年之后,对于Apple App来说,如果要支持第三方登录,则必须同时支持苹果的第三方登录,即Sign in With Apple, 本文主要介绍如何使用Go语言实现Sign in Wit ...

  4. iOS内购 - 服务端票据验证及漏单引发的思考

    因业务需要实现了APP内购处理,但在过程中出现了部分不可控的因素,导致部分用户反映有充值不成并漏单的情况. 仔细考虑了几个付费安全上的问题,凡是涉及到付费的问题都很敏感,任何一方出现损失都是不能接受的 ...

  5. IOS苹果内购 PHP后端验证票据

    大体流程: 1.IOS端需要在iTunes Connect上面添加配置一些内购商品,并审核通过,每个内购商品有自己的唯一标识product_id. 2.PHP后端要有一套与之对应的内购商品.IOS应用 ...

  6. php 苹果支付验证,IOS苹果内购 PHP后端验证票据

    大体流程: 1.IOS端需要在iTunes Connect上面添加配置一些内购商品,并审核通过,每个内购商品有自己的唯一标识product_id. 2.PHP后端要有一套与之对应的内购商品.IOS应用 ...

  7. 苹果授权登陆 服务端验证(java)

    需要的jar包: jose4j-0.6.4.jar:jwks-rsa-0.9.0.jar:jjwt-0.9.1.jar: jar包下载地址:https://download.csdn.net/down ...

  8. ios内购二次验证安全性问题_苹果IOS内购二次验证返回state为21002的坑

    项目是三四年前的老项目,之前有IOS内购二次验证的接口,貌似很久都没用了,然而最近IOS的妹子说接口用不了,让我看看啥问题.接口流程时很简单的,就是前端IOS在购买成功之后,接收到receipt后进行 ...

  9. go token验证_GitHub - goflyfox/gtoken: 基于gf框架的token插件,通过服务端验证方式实现token认证;...

    gtoken 介绍 基于GoFrame框架的token插件,通过服务端验证方式实现token认证:已完全可以支撑线上token认证,通过Redis支持集群模式:使用简单,大家可以放心使用: gtoke ...

最新文章

  1. sqlite在c++中的使用方法
  2. 单例模式 之 单例模式——枚举
  3. angularjs 获取复选框的值_哈迪斯仙酒有什么用 哈迪斯仙酒获取方式以及作用一览...
  4. 使用Reactor进行反应式编程最全教程
  5. mysql 不支持 select into
  6. Windows-server-2008-R2安装Oracle-11g-R2-dataguard
  7. 深入掌握JMS(二):一个JMS例子
  8. INET Sockets Tour: ioctl()
  9. mysql查询单表的销售额_MYsql数据库单表百万数据量查询
  10. docker数据圈_《Docker从入门到跑路》之镜像和容器的基本操作
  11. 移动端overflow-x去掉滑动条
  12. 识别视频文件夹,listview
  13. android 开源gis,开源GIS之--移动GIS
  14. mysql是正排还是倒排_正排索引与倒排索引的理解
  15. DDOS攻击是什么意思?服务器怎么防DDOS攻击?
  16. C# 解析CSV文件
  17. 有没有人用过迅捷PDF转换器?文件转换效果怎么样?
  18. CentOS7.3下Zabbix3.5之微信报警配置
  19. win7母机上的vmware12中Ubuntu16中安装复制粘贴工具
  20. 各种文件后缀名对应content-type

热门文章

  1. WeekHashMap
  2. 算法练习day1——190318(二分查找)
  3. UML 类图几种关系的总结
  4. Qt5 常见的控件类关系
  5. 小米10的Android安全更新,MIUI 12首批更新名单被曝光,小米10系列优先上Android 11...
  6. getValue()方法 java_java.util.zip.CRC32.getValue()方法示例
  7. mysql 二进制日志详解_Mysql二进制日志详解
  8. php html url编码,html中url编码是什么?有什么用?
  9. 超过200m文件发送_苦等10年的微信功能——大文件功能来了
  10. Qt: QTimer和QThread