当用户在使用应用内购买功能的时候,若是用户支付成功了,因为网络或者其它不可预计的因素,致使APP应用没有将相应的商品或服务提供给用户。不管APP仍是用户,都不想看到,所以,这是不容许发生的状况,要由程序去控制稳定性。网络
由此,便引出了这次讨论的话题,iOS应用内购买IAP的支付凭证验证,失败后的重试机制。dom
整个流程主要分为如下两步。atom
首次验证失败后,根据设定的重试次数,当即重试指定次数。spa
若是当即重试指定次数仍是失败,则进入第二步。code
当即重试后仍是失败,则使用本地文件的方式,保存订单等相关支付信息。orm
具体方式是,将必要的信息存入NSDictionary,而后保存为plist文件。plist文件的命名,最好包含用户ID、订单ID,和其它具备惟一性的字符,以区分不一样用户,不一样的订单等。进程
2.1步的保存文件的代码ip
/** 存储用户购买凭证 @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]; }
保存信息完成后,则间隔指定时间不断重试,直到验证成功,或者APP进程结束。get
间隔时间,可自定义,我的以为,5分钟以上的间隔,会比较合适。string
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); } }
在启动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]); } }
在每次重试时,验证成功后,须要将本地存储的文件移除,防止重复验证。