JAVAIOS内购二次验证及掉单问题解决

Posted 现世安稳。

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JAVAIOS内购二次验证及掉单问题解决相关的知识,希望对你有一定的参考价值。

这个估计是我踩过的最大的坑,当时做微信支付的时候也没这么坑爹,当然他俩也半斤八两。。。

 

苹果官方明确表示:验证支付时,可能会有一定的延迟。第一次处理的时间就专注的解决这个问题了,忽略了掉单的问题(稍后再说),让我多次更新支付代码才降低了掉单率。

常识

1,返回状态码含义

2、正常返回结果格式

{
  "environment": "Sandbox",
  "receipt": {
    "in_app": [
      {
        "transaction_id": "10000004111119001",
        "original_purchase_date": "2018-07-06 03:16:41 Etc/GMT",
        "quantity": "1",
        "original_transaction_id": "1000000414619001",
        "purchase_date_pst": "2018-07-05 20:16:41 America/Los_Angeles",
        "original_purchase_date_ms": "1530847001000",
        "purchase_date_ms": "1530847001000",
        "product_id": "com.Beixxxxxxxxxon.fourc",
        "original_purchase_date_pst": "2018-07-05 20:16:41 America/Los_Angeles",
        "is_trial_period": "false",
        "purchase_date": "2018-07-06 03:16:41 Etc/GMT"
      }
    ],
    "adam_id": 0,
    "receipt_creation_date": "2018-07-06 03:16:41 Etc/GMT",
    "original_application_version": "1.0",
    "app_item_id": 0,
    "original_purchase_date_ms": "1375340400000",
    "request_date_ms": "1530847558058",
    "original_purchase_date_pst": "2013-08-01 00:00:00 America/Los_Angeles",
    "original_purchase_date": "2013-08-01 07:00:00 Etc/GMT",
    "receipt_creation_date_pst": "2018-07-05 20:16:41 America/Los_Angeles",
    "receipt_type": "ProductionSandbox",
    "bundle_id": "com.jiaxxxxxmei.www.Gxxxxxxxrooms",
    "receipt_creation_date_ms": "1530847001000",
    "request_date": "2018-07-06 03:25:58 Etc/GMT",
    "version_external_identifier": 0,
    "request_date_pst": "2018-07-05 20:25:58 America/Los_Angeles",
    "download_id": 0,
    "application_version": "3"
  },
  "status": 0
}
View Code

3、凭证(此处提供的是测试环境的成功支付凭证)

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
View Code

 

4、测试|正式验证地址

//购买凭证验证地址  
private static final String certificateUrl = "https://buy.itunes.apple.com/verifyReceipt";

//测试的购买凭证验证地址
private static final String certificateUrlTest = "https://sandbox.itunes.apple.com/verifyReceipt";

 


 

开始JAVA开发接口验证:

初步代码,第一版,实现正常的支付功能(其中充值金额最好从返回结果中获取,不过我是让前端传过来的):

     /** 
     * 接收ios端发过来的购买凭证
     * @param
     */  
    @RequestMapping("/setIapCertificate")
    @ResponseBody
    public R setIapCertificate(HttpServletRequest req){
        String certificateCode = req.getParameter("certificateCode");
        String money = req.getParameter("money");//充值金额
        String type = req.getParameter("type");//1、正式  2、测试
        String url = null;  
        url = type.equals("1")?certificateUrl : certificateUrlTest;  //
       
        if(StringUtils.isNotEmpty(certificateCode)){  
            String r = sendHttpsCoon(url, certificateCode); //向苹果服务器发送凭证并接受结果 
            JSONObject params = JSONObject.parseObject(r);
            System.out.println("*****************"+params);
            String code = "";
            String environment = "";
            String transaction_id = "";
            final String orderNum = DateUtil.getRechargeOrderNum();//根据当前时间生成订单号,yyMMddHHmmssSSS + 6位随机数
            if(params != null) {
                code = String.valueOf(params.get("status"));
                if("0".equals(code)) {//成功
                    environment = String.valueOf(params.get("environment"));
                    String r_receipt = params.getString("receipt");
                    JSONObject returnJson = JSONObject.parseObject(r_receipt);
                    System.out.println("*****************"+returnJson);
                    String in_app = returnJson.getString("in_app");

                    if (in_app.endsWith("]")) {
                        JSONObject in_appJson = JSONObject.parseObject(in_app.substring(1, in_app.length()-1));
                        transaction_id = in_appJson.getString("transaction_id");   // 订单号
                    }
                    //保存充值记录
                    saveRecharge(orderNum, money, environment, transaction_id);
                    return R.ok();
                }
            }else{
                return R.error();
            }
        }else{  
            return R.error(); 
        }  
    }  
      
View Code

 

后来发现大部分可以支付成功,但部分充值不成功,考虑可能是网络的问题,然后利用定时任务实现多次请求。若第一次请求结果不是成功状态则保存第一次请求的结果,并且每隔10秒请求一次,最多请求6次(一分钟)。

就有了优化后的第二版

 /**
     * 接收iOS端发过来的购买凭证(方法已过时,被替代)
     * @param
     */
    @RequestMapping("/setIapCertificate1")
    @ResponseBody
    public R setIapCertificate1(HttpServletRequest req){
        String certificateCode = req.getParameter("certificateCode");
        String money = req.getParameter("money");//充值金额
        String type = req.getParameter("type");//1、正式  2、测试
        String url = null;
        url = type.equals("1")?certificateUrl : certificateUrlTest;  //

        if(StringUtils.isNotEmpty(certificateCode)){
            String r = sendHttpsCoon(url, certificateCode);
            JSONObject params = JSONObject.parseObject(r);
            System.out.println("*****************"+params);
            String code = "";
            String environment = "";
            String transaction_id = "";
            final String orderNum = DateUtil.getRechargeOrderNum();//根据当前时间生成订单号,yyMMddHHmmssSSS + 0~1000000随机数
            if(params != null) {

                code = String.valueOf(params.get("status"));

                payError = new WfIospayErrorEntity();
                payError.setCertificatecode(certificateCode);
                payError.setMoney(money);
                payError.setRebate(rebate);
                payError.setResultcode(code);
                payError.setType(type);
                payError.setUpdatetime(new Date());
                payError.setRemark(params.toString());
                iospayErrorService.save(payError);
                
                if("0".equals(code)) {//成功
                    environment = String.valueOf(params.get("environment"));
                    String r_receipt = params.getString("receipt");
                    JSONObject returnJson = JSONObject.parseObject(r_receipt);
                    System.out.println("*****************"+returnJson);
                    String in_app = returnJson.getString("in_app");

                    if (in_app.endsWith("]")) {
                        JSONObject in_appJson = JSONObject.parseObject(in_app.substring(1, in_app.length()-1));
                        transaction_id = in_appJson.getString("transaction_id");   // 订单号
                    }
                    //保存充值记录
                    saveRecharge(rebate, orderNum, uid, money, environment, transaction_id);
                    return R.ok();
                }else{//失败

                    final String url1 = url;
                    final String certificateCode1 = certificateCode;
                    final String environment1 = environment;
                    final String transaction_id1 = transaction_id;
                    final String money1 = money;//充值金额
                    final Timer timer = new Timer();
                    timer.schedule(new TimerTask() {//若失败,则每10请求一次

                        @Override
                        public void run() {
                            if (map.get(certificateCode1) != null) {
                                map.put(certificateCode1, map.get(certificateCode1) + 1);
                            } else {
                                map.put(certificateCode1, 1);
                            }
                            if (map.get(certificateCode1) >= 60) {
                                map.remove(certificateCode1);
                                timer.cancel();
                            }
                            String r1 = sendHttpsCoon(url1, certificateCode1);
                            if (JSONObject.parseObject(r1).getString("status").equals("0")) {
                                saveRecharge( orderNum, money1, environment1, transaction_id1);//当成功时处理充值业务
                                map.remove(certificateCode1);
                                timer.cancel();
                            }
                            //System.out.println("设定指定任务task在指定延迟delay后执行");
                        }
                    },1000, 10000);

                    return R.error("68");
                }
            }else{//保存失败凭证
                payError = new WfIospayErrorEntity();
                payError.setCertificatecode(certificateCode);
                payError.setMoney(money);
                payError.setRebate(rebate);
                payError.setResultcode(code);
                payError.setType(type);
                payError.setUpdatetime(new Date());
                if(params != null){
                    payError.setRemark("params为空");
                }else {
                    payError.setRemark(params.toString());
                }
                iospayErrorService.save(payError);
                return R.error();
            }

        }else{
            return R.error();
        }
    }
View Code

 

but,还是有掉单的问题,此时我崩溃了,想不明白怎么回事。直到我查了苹果返回的结果发现,根据一个凭证查出了两次支付的订单信息

{
  "environment": "Production",
  "receipt": {
    "in_app": [
      {
        "transaction_id": "470000347480489",
        "original_purchase_date": "2018-07-13 06:01:00 Etc/GMT",
        "quantity": "1",
        "original_transaction_id": "470000347480489",
        "purchase_date_pst": "2018-07-12 23:01:00 America/Los_Angeles",
        "original_purchase_date_ms": "1531461660000",
        "purchase_date_ms": "1531461660000",
        "product_id": "com.Beijxxxxxxxxation.fourc",
        "original_purchase_date_pst": "2018-07-12 23:01:00 America/Los_Angeles",
        "is_trial_period": "false",
        "purchase_date": "2018-07-13 06:01:00 Etc/GMT"
      },
      {
        "transaction_id": "470000347479507",
        "original_purchase_date": "2018-07-13 05:56:51 Etc/GMT",
        "quantity": "1",
        "original_transaction_id": "470000347479507",
        "purchase_date_pst": "2018-07-12 22:56:51 America/Los_Angeles",
        "original_purchase_date_ms": "1531461411000",
        "purchase_date_ms": "1531461411000",
        "product_id": "com.Beixxxxxxxxxxxcation.fourb",
        "original_purchase_date_pst": "2018-07-12 22:56:51 America/Los_Angeles",
        "is_trial_period": "false",
        "purchase_date": "2018-07-13 05:56:51 Etc/GMT"
      }
    ],
    "adam_id": 1375992347,
    "receipt_creation_date": "2018-07-13 06:01:00 Etc/GMT",
    "original_application_version": "3",
    "app_item_id": 1375992347,
    "original_purchase_date_ms": "1531016326000",
    "request_date_ms": "1531461663282",
    "original_purchase_date_pst": "2018-07-07 19:18:46 America/Los_Angeles",
    "original_purchase_date": "2018-07-08 02:18:46 Etc/GMT",
    "receipt_creation_date_pst": "2018-07-12 23:01:00 America/Los_Angeles",
    "receipt_type": "Production",
    "bundle_id": "com.jxxxxxxxnmei.www.Goxxxxxxsrooms",
    "receipt_creation_date_ms": "1531461660000",
    "request_date": "2018-07-13 06:01:03 Etc/GMT",
    "version_external_identifier": 827702473,
    "request_date_pst": "2018-07-12 23:01:03 America/Los_Angeles",
    "download_id": 87033802813938,
    "application_version": "3"
  },
  "status": 0
}
View Code

知道原因后,想解决办法。读取苹果服务器返回的订单数组,循环充值。但这样的话,需要避免重复充值,所以现在需要用缓存给每次充值做个记录(我选择的redis),在充值之前判断苹果服务器返回的订单中有没有已经充过值的。

然后有了第三版

 /**
     * 接收iOS端发过来的购买凭证
     *
     * @param
     */
    @RequestMapping("/setIapCertificate")
    @ResponseBody
    public R setIapCertificate(HttpServletRequest req) {
        final Integer uid = WebUtils.getUid(req);
        String certificateCode = req.getParameter("certificateCode");
        String money = req.getParameter("money");//充值金额
        String type = req.getParameter("type");//1、正式  2、测试
        String url = null;
        url = type.equals("1") ? certificateUrl : certificateUrlTest;  //

        WfIospayErrorEntity payError = new WfIospayErrorEntity();
        payError.setCertificatecode(certificateCode);
        payError.setMoney(money);
        payError.setRebate(rebate);
        payError.setResultcode("110");
        payError.setType(type);
        payError.setUid(String.valueOf(uid));
        payError.setUpdatetime(new Date());
        payError.setRemark("params为空");
        iospayErrorService.save(payError);

        if (StringUtils.isNotEmpty(certificateCode)) {
            String r = sendHttpsCoon(url, certificateCode);
            JSONObject params = JSONObject.parseObject(r);
            System.out.println("*****************" + params);
            String code = "";
            String environment = "";
            String transaction_id = "";
            final String orderNum = DateUtil.getRechargeOrderNum();//根据当前时间生成订单号,yyMMddHHmmssSSS + 0~1000000随机数
            if (params != null) {

                code = String.valueOf(params.get("status"));

                payError = new WfIospayErrorEntity();
                payError.setCertificatecode(certificateCode);
                payError.setMoney(money);
                payError.setRebate(rebate);
                payError.setResultcode(code);
                payError.setType(type);
                payError.setUid(String.valueOf(uid));
                payError.setUpdatetime(new Date());
                payError.setRemark(params.toString());
                iospayErrorService.save(payError);


                if ("0".equals(code)) {//成功
                    environment = String.valueOf(params.get("environment"));
                    String r_receipt = params.getString("receipt");
                    JSONObject returnJson = JSONObject.parseObject(r_receipt);
                    System.out.println("*****************" + returnJson);
                    //String in_app = returnJson.getString("in_app");
                    JSONArray jsonArray = (JSONArray) returnJson.get("in_app");// 获取返回结果中的订单列表
                    JSONObject targetOrder = null;
                    String product_id = null;
                    for (int i = 0; i < jsonArray.size(); i++) {

                        targetOrder = jsonArray.getJSONObject(i);//获取订单信息对象
                        product_id = targetOrder.getString("product_id");//获取产品信息
                        transaction_id = targetOrder.getString("transaction_id");// transaction_id交易号

                        if (!ISsoLoginHelper.confirmPay(uid, transaction_id)) {//redis验证订单是否已经充值
                            //保存充值记录
                            ISsoLoginHelper.savePay(uid, transaction_id, money);//用redis保存记录
                            //进行充值
                            saveRecharge(rebate, orderNum, uid, money, environment, transaction_id);
                        }
                    }

                    return R.ok();
                } else {//失败

                    final String url1 = url;
                    final String certificateCode1 = certificateCode;
                    final String environment1 = environment;
                    final String transaction_id1 = transaction_id;
                    final String money1 = money;//充值金额
                    final Timer timer = new Timer();
                    timer.schedule(new TimerTask() {

                        @Override
                        public void run() {
                            if (map.get(certificateCode1) != null) {
                                map.put(certificateCode1, map.get(certificateCode1) + 1);
                            } else {
                                map.put(certificateCode1, 1);
                            }
                            if (map.get(certificateCode1) >= 60) {
                                map.remove(certificateCode1);
                                timer.cancel();
                            }
                            String r1 = sendHttpsCoon(url1, certificateCode1);
                            if (JSONObject.parseObject(r1).getString("status").equals("0")) {
                                if (!ISsoLoginHelper.confirmPay(uid, transaction_id1)) {//redis验证订单是否已经充值
                                    //保存充值记录
                                    ISsoLoginHelper.savePay(uid, transaction_id1, money1);
                                    //进行充值
                                    saveRecharge(rebate, orderNum, uid, money1, environment1, transaction_id1);
                                }
                                map.remove(certificateCode1);
                                timer.cancel();
                            }
                            //System.out.println("设定指定任务task在指定延迟delay后执行");
                        }
                    }, 1000, 10000);

                    return R.error("68");
                }
            } else {

                payError = new WfIospayErrorEntity();
                payError.setCertificatecode(certificateCode);
                payError.setMoney(money);
                payError.setRebate(rebate);
                payError.setResultcode("111");
                payError.setType(type);
                payError.setUid(String.valueOf(uid));
                payError.setUpdatetime(new Date());
                payError.setRemark("params再次为空");
                iospayErrorService.save(payError);

                return R.error();
            }

        } else {
            return R.error();
        }
    }
View Code

 

 


 

其中涉及的工具或方法:

 

发送请求:

      
    /** 
     * 重写X509TrustManager 
     */  
    private static TrustManager myX509TrustManager = new X509TrustManager() {  
          
        @Override  
        public X509Certificate[] getAcceptedIssuers() {  
            return null;  
        }  
          
        @Override  
        public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {  
              
        }  
          
        @Override  
        public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {  
              
        }  
    };


  /** 
     * 发送请求 
     * @param url 
     * @param
     * @return 
     */  
    private String sendHttpsCoon(String url, String code){  
        if(url.isEmpty()){  
            return null;  
        }  
        try {  
            //设置SSLContext  
            SSLContext ssl = SSLContext.getInstance("SSL");  
            ssl.init(null, new TrustManager[]{myX509TrustManager}, null);  
              
            //打开连接  
            HttpsURLConnection conn = (HttpsURLConnection) new URL(url).openConnection();  
            //设置套接工厂  
            conn.setSSLSocketFactory(ssl.getSocketFactory());  
            //加入数据  
            conn.setRequestMethod("POST");  
            conn.setDoOutput(true);  
            conn.setRequestProperty("Content-type","application/json");  
              
            JSONObject obj = new JSONObject();  
            obj.put("receipt-data", code);  
              
            BufferedOutputStream buffOutStr = new BufferedOutputStream(conn.getOutputStream());  
            buffOutStr.write(obj.toString().getBytes());  
            buffOutStr.flush();  
            buffOutStr.close();  
              
            //获取输入流  
            BufferedReader reader = new BufferedReader(new InputStreamReader(conn.getInputStream()));  
              
            String line = null;  
            StringBuffer sb = new StringBuffer();  
            while((line = reader.readLine())!= null){  
                sb.append(line);  
            }  
            return sb.toString();  
          
        } catch (Exception e) {  
            return null;  
        }  
    }  
View Code

redis保存|判断充值记录(安装配置我就不多说了,不懂的请自行百度)

    /**
     * 保存ios订单支付信息
     *
     * @param userid
     * @return
     */
    public static void savePay(Integer userid,String transaction_id,String money) {
        Jedis jedis = myJedisPool.getResource();

        try {
            jedis.setnx(userid +"_"+ transaction_id , money);
        } catch (Exception e) {
            e.printStackTrace();
            logger.error(e.getMessage(), e);
        } finally {
            jedis.close();
        }
    }

    /**
     * 判断订单是否已充值成功
     *
     * @param userid
     * @return
     */
    public static boolean confirmPay(Integer userid,String transaction_id) {
        Jedis jedis = myJedisPool.getResource();
        boolean b = false;
        try {
            b = jedis.exists(userid +"_"+ transaction_id);
        } catch (Exception e) {
            e.printStackTrace();
            logger.error(e.getMessage(), e);
        } finally {
            jedis.close();
        }
        return b;
    }
View Code

 

以上是关于JAVAIOS内购二次验证及掉单问题解决的主要内容,如果未能解决你的问题,请参考以下文章

ios IAP 内购验证

Keep客户端 In-App Purchase 掉单踩坑指南

Keep客户端 In-App Purchase 掉单踩坑指南

Keep客户端 In-App Purchase 掉单踩坑指南

Keep客户端 In-App Purchase 掉单踩坑指南

Flutter iOS内购(代码篇-全网最全)