Ios内购防漏单、刷单、订单重复

Ios苹果支付流程:

1、客户端先从苹果获取内购Id。
2、客户端将内购id,金额、用户id等传给服务端获取一个自己服务端生成的订单号。
3、客户端向苹果发起支付。
4、支付成功后,客户端从本地拿支付凭证、将支付凭证和订单号、用户id等参数传给服务端;服务端拿支付凭证向苹果发起验证订单是否有效,然后将结果反馈给客户端。
5、客户端刷新用户个人信息。

漏单:
以上流程中可能会出现漏单的情况:当客户端向苹果支付成功后,准备向服务端发起验证,此时若网络发生抖动或者人为退出app等情况,就会出现用户充值成功,但是没加金币的情况。
解决方案:在paymentQueue:updatedTransactions:代理方法中获取到支付成功的回调,此时马上缓存订单信息(包括支付凭证、订单号、用户id等;担心用户卸载app的,可以直接缓存到钥匙串),然后再向服务端验证订单。若订单验证成功,则删除该项订单缓存;否则不做处理。另外,每次app启动的时候,可以延时几秒钟,去检测本地是否有未验证成功的苹果订单;如果有,就一个个重新验证。

刷单:
我们公司的app最早的时候,服务端还是用的Http请求,也没有考虑到刷单这一块,然后被某些用户钻了空子;他们先发起苹果支付,获取一个订单号,然后拿这个订单号和之前支付成功的凭证,向服务端发起验证;然后服务端发现订单号也对得上,凭证也验证有效,就给用户加金币了;用户用同一个凭证不断的换订单号。。。。。
解决方案:首先服务端每次生成的订单号都缓存起来,客户端在请求完订单号后,将该订单号记录在内购订单对象里(赋值给SKPayment的applicationUsername属性);然后当客户端支付成功后,把订单id和支付凭证传给服务端,服务端先拿凭证去苹果校验,苹果返回的结果中有一个transactionId(跟前面记录的applicationUsername是一致的) ,这时候服务端判断该transactionId和订单id是否一致,若一致,且该订单id是一个新id,之前没有缓存过;才算是完全的校验成功。

订单重复:
订单重复的情况就是支付成功后,从本地拿到了错误的支付凭证(之前的凭证),去校验的时候,会发现该订单已经完成了。
我这边发现这个问题的情况是这样的:客户端发起苹果支付的时候调用addPayment方法,在paymentQueue:updatedTransactions:回调中获取支付的状态,支付成功、支付失败和重复购买的状态下要结束订单finishTransaction。我当时是支付失败的情况,没有结束订单,然后下次支付成功的时候,获取的支付凭证是上次失败的凭证,然后不仅校验会失败,订单也是重复的。

  • 6
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是 PHP 代码实现 iOS 内购订单验证的示例: ```php <?php // 定义 app id 和 app secret $app_id = 'your_app_id'; $app_secret = 'your_app_secret'; // 获取验证请求参数 $receipt = $_POST['receipt']; $password = $_POST['password']; // 发送验证请求到 Apple 服务器 $url = 'https://buy.itunes.apple.com/verifyReceipt'; // 正式环境 //$url = 'https://sandbox.itunes.apple.com/verifyReceipt'; // 沙盒环境 $data = json_encode(array('receipt-data' => $receipt, 'password' => $password)); $options = array( 'http' => array( 'header' => "Content-type: application/json\r\n", 'method' => 'POST', 'content' => $data ) ); $context = stream_context_create($options); $result = file_get_contents($url, false, $context); // 解析验证结果 $response = json_decode($result, true); if ($response['status'] == 0) { // 验证成功 $latest_receipt_info = $response['latest_receipt_info']; $transaction_id = $latest_receipt_info[0]['transaction_id']; $product_id = $latest_receipt_info[0]['product_id']; $purchase_date = $latest_receipt_info[0]['purchase_date']; // TODO: 在这里处理订单验证成功后的逻辑 } else { // 验证失败 $error_code = $response['status']; $error_message = $response['status_message']; // TODO: 在这里处理订单验证失败后的逻辑 } ?> ``` 以上代码中,首先定义了 app id 和 app secret,然后获取了验证请求参数 receipt 和 password,接着发送验证请求到 Apple 服务器,并解析验证结果。如果验证成功,可以获取到订单的相关信息,如 transaction_id、product_id 和 purchase_date 等,可以在这里处理订单验证成功后的逻辑;如果验证失败,可以获取到错误码和错误信息,可以在这里处理订单验证失败后的逻辑。需要注意的是,验证请求需要发送到正式环境或者沙盒环境,根据实际情况选择相应的 URL 即可。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值