iOS应用内购买IAP的支付凭证验证失败后的重试机制
当用户在使用应用内购买功能的时候,如果用户支付成功了,由于网络或者其它不可预计的因素,导致APP应用没有将相应的商品或服务提供给用户。无论APP还是用户,都不想看到,因此,这是不允许发生的情况,要由程序去控制稳定性。
由此,便引出了此次讨论的话题,iOS应用内购买IAP的支付凭证验证,失败后的重试机制。
整个流程主要分为以下两步。
1 首次验证失败后的立即重试
首次验证失败后,根据设定的重试次数,立即重试指定次数。
如果立即重试指定次数还是失败,则进入第二步。
2 立即重试后还是失败的情况处理
2.1 本地文件保存订单等相关支付信息
立即重试后还是失败,则使用本地文件的方式,保存订单等相关支付信息。
具体方式是,将必要的信息存入NSDictionary,然后保存为plist文件。plist文件的命名,最好包含用户ID、订单ID,和其它具有唯一性的字符,以区分不同用户,不同的订单等。
2.1步的保存文件的代码
/**
存储用户购买凭证
@param receipt 购买凭证
@param sID 唯一标识(比如UserId)
@param orderNum 订单号
*/
+ (void)saveReceiptValidation:(NSString *_Nonnull)receipt
withID:(NSString *_Nonnull)sID
orderNum:(NSString *_Nonnull)orderNum
{
NSDate *dateSaved = [NSDate date];
NSString *fileName = [NSString stringWithFormat:@"IAPInfo-%@-%@", sID, orderNum];
NSString *fileDir = [[self class] getIAPInfoLocalFilePath:sID];
NSString *savedPath = [NSString stringWithFormat:@"%@%@.plist", fileDir, fileName];
NSDictionary *dic =[NSDictionary dictionaryWithObjectsAndKeys:
receipt, kReceipStringKey,
dateSaved, kReceipDateKey,
sID, kReceipIdKey,
orderNum, kOrderNumKey,
nil];
NSFileManager *fileManager = [NSFileManager defaultManager];
BOOL isDir = FALSE;
BOOL isDirExist = [fileManager fileExistsAtPath:fileDir isDirectory:&isDir];
if(!(isDirExist && isDir)) {//目录不存在
BOOL bCreateDir = [fileManager createDirectoryAtPath:fileDir withIntermediateDirectories:YES attributes:nil error:nil];
if(!bCreateDir){
NSLog(@"Create Directory Failed.");
} else {
[[self class] saveFile:savedPath withDictionary:dic];
}
} else {//目录存在,直接保存
[[self class] saveFile:savedPath withDictionary:dic];
}
}
+ (BOOL)saveFile:(NSString *)savedPath withDictionary:(NSDictionary *)dic
{
BOOL isWrited = [dic writeToFile:savedPath atomically:YES];
NSLog(@"saveReceiptValidation is success ? %@, at savedPath:%@", @(isWrited), savedPath);
return isWrited;
}
+ (NSString *)getIAPInfoLocalFilePath:(NSString *)sID
{
return [NSString stringWithFormat:@"%@/IAPReceipt-%@/", [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject], sID];
}
2.2 间隔指定时间不断重试
保存信息完成后,则间隔指定时间不断重试,直到验证成功,或者APP进程结束。
间隔时间,可自定义,个人觉得,5分钟以上的间隔,会比较合适。
1步和2步的重试逻辑的部分代码
if (retried < kRetryMax) {
//重试
[[self class] validateReceipt:receipt orderNum:orderNum retriedTimes:retried+1 success:success failure:failure];
} else {
//重试了kRetryMax次后,还失败,则创建延时任务,5分钟后重试
int afterTime = 5 * 60;
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(afterTime * NSEC_PER_SEC)), dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
[[self class] validateReceipt:receipt orderNum:orderNum retriedTimes:0 success:nil failure:nil];
});
//保存凭证
NSString *userId = [TTVUserInfo sharedTTVUserInfo].currentUser.userId;
if (userId && userId.length > 0) {
[[self class] saveReceiptValidation:receipt withID:userId orderNum:orderNum];
}
//错误回调
if (failure) {
failure(errCode, errMessage);
}
}
3 在APP进程结束前,都未验证成功的情况处理
3.1 在启动或登录成功时重新发起验证
在启动APP时,如果用户已经登录,则将所有验证失败的支付凭证重新进行验证。如果用户未登录,则订阅通知,在用户登录首次登录成功后,重新发起验证流程。
3步的重新发起验证流程的部分代码
/**
验证receipt失败,再次验证
@param sID 唯一标识(比如UserId)
*/
+ (void)resendFailedReceiptValidation:(NSString *)sID
{
NSFileManager *fileManager = [NSFileManager defaultManager];
NSError *error = nil;
NSString *filePath = [[self class] getIAPInfoLocalFilePath:sID];
//搜索该目录下的所有文件和目录
NSArray *cacheFileNameArray = [fileManager contentsOfDirectoryAtPath:filePath error:&error];
NSLog(@"resendFailedReceiptValidation has files : %@", cacheFileNameArray);
if (error == nil)
{
for (NSString *name in cacheFileNameArray)
{
if ([name hasSuffix:@".plist"])//如果有plist后缀的文件,说明就是存储的购买凭证
{
NSString *plistPath = [NSString stringWithFormat:@"%@/%@", filePath, name];
[[self class] resendValidationRequest:plistPath];
}
}
}
else
{
NSLog(@"getIAPInfoLocalFilePath error:%@", [error domain]);
}
}
注意事项
在每次重试时,验证成功后,需要将本地存储的文件移除,防止重复验证。