zoukankan      html  css  js  c++  java
  • 【JAVA】IOS内购二次验证及掉单问题解决

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

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

    常识:

    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
  • 相关阅读:
    在SharePoint 2010中创建网站的权限级别
    SharePoint 2013 Pop-Up Dialogs
    SharePoint 2010 Pop-Up Dialogs
    sharepoint 2010 页面添加footer方法 custom footer for sharepoint 2010 master page
    Using SharePoint 2010 dialogs
    Spring Security
    mysql优化
    memcached缓存技术
    网页静态化技术
    最小生成树
  • 原文地址:https://www.cnblogs.com/hero123/p/9324779.html
Copyright © 2011-2022 走看看