Apple 內購流程:
1、向蘋果服務器,發送請求,獲取可購買商品信息。
2、SKProductsRequest 協議 。獲取商品數據列表。
3、確定購買商品(或回復商品)。
4、監測交易過程。
5、驗證交易收據(receipt)。
1、向蘋果服務器,發送請求,獲取可購買商品信息。
//提交商品信息請求
- (void)requestProductWithProductArray:(NSArray *)productIdAry {
NSSet *set = [[NSSet alloc] initWithArray:productIdAry];
//“異步”請求有哪些可售商品
SKProductsRequest *request = [[SKProductsRequest alloc] initWithProductIdentifiers:set];
//啟動服務
[request start];
}
2、SKProductsRequest 協議 。獲取商品數據列表。
#pragma mark - SKProductsRequest Delegate
- (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response {
//獲取商品數據
NSArray *productAry = response.products;
if ([productAry count] <= 0) {
//暫無商品信息
return;
}
//初始化本地商品信息個數
if (self.productDic == nil) {
self.productDic = [NSMutableDictionary dictionaryWithCapacity:response.products.count];
}
//遍歷商品數據
for (SKproduct *product in response.products) {
[self.productDis setObject:product forKey:product.productIdentifier];
//打印產品列表信息
NSLog(@"product ID : %@",product.productIdentifier);
NSLog(@"product Description : %@",product.description);
NSLog(@"product Title :%@",product.localizedTitle);
}
#warning - Todo 可以將商品信息列表傳給ViewController以供顯示商品列表。
#warning - Todo 但不建議使用此信息顯示商品列表,可以用此信息進行產品ID的驗證。
}
#pragma mark - SKProductsRequest Delegate
- (void)request:(SKRequest *)request didFailWithError:(NSError *)error {
NSLog(@"---------------- 錯誤 ---------------");
}
#pragma mark - SKProductsRequest Delegate
- (void)requestDidFinish:(SKPRequest *)request {
NSLog(@"--------------- 反饋信息結束 ---------------");
}
3、確定購買商品
#pragma mark - Buy Product
- (void)buyProduct:(NSString *)productID {
//購買商品
SKProduct *buyProduct = self.productDic[productID];
SKPayment *payment = [SKPayment paymentWithProduct:buyProduct];
[[SKPaymentQueue defaultQueue] addPayment:payment];
}
4、回復商品
#pragma mark - Restore Product
- (void)restorePurchase {
//回復已經完成的所有交易(僅限永久有效商品)。
[[SKPaymentQueue defaultQueue] restoreCompletedTransactions];
}
5、監測交易過程
#pragma mark - SKPaymentTransaction Observer
- (void)paymentQueue:(SKPaymentQueue *)queue updateTransactions:(NSArray<SKPaymentTransaction *> *)transaction {
for (SKPaymentTransaction *transaction in transactions) {
NSLog(@"交易隊列狀態 :%@",transactions);
switch (transaction.transactionState) {
case SKPaymentTransactionStatePurchased: //購買成功
// 向蘋果服務器進行驗證
if (self.checkAfterPay) {
//支付成功了,并開始向蘋果服務器進行驗證(若CheckAfterPay為NO,則不會經過此步驟)
}else {
//商品完全購買成功且驗證成功了。(若CheckAfterPay為NO,則會在購買成功后直接觸發此方法)
}
// 目前都是全部進行驗證。
[self verifyPruchaseWithPaymentTransaction:transaction isTestServer:NO];
break;
case SKPaymentTransactionStatePurchasing: // 正在購買
NSLog(@"正在購買...");
break;
case SKPaymentTransactionStateRestored: // 恢復成功
// 將交易從交易隊列中刪除
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
break;
case SKPaymentTransactionStateFailed: // 購買取消或失敗
if (transaction.error.code != SKErrorPaymentCancelled) {
NSLog(@"購買失敗!");
}else{
NSLog(@"購買取消!");
}
// 將交易從交易隊列中刪除
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
break;
default:
// 將交易從交易隊列中刪除
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
break;
}
}
}
6、驗證交易收據(receipt)
#pragma mark - Verify Receipt
- (void)verifyPruchaseWithPaymentTransaction:(SKPaymentTransaction *)transaction isTestServer:(BOOL)flag {
// 驗證憑據,獲取到蘋果返回的交易憑據
// appStore Receipt URL iOS7.0增加的,購買交易完成后,會將憑據存放在該地址
NSURL *receiptURL = [[NSBundle mainBundle] appStoreReceiptURL];
// 從沙盒中獲取到購買憑據receipte
NSData *receipt = [NSData dataWithContentsOfURL:receiptURL];
if (!receipt) {
// 交易憑證為空,驗證失敗
// KIAPPurchVerFailed
return;
}else {
// 購買成功將交易憑證發送給服務端進行再次校驗
// kIAppurchSuccess
}
// 在網絡中傳輸數據,大多情況下是傳輸的字符串而不是二進制數據
// 傳輸的是BASE64編碼的字符串
/**
BASE64 常用的編碼方案,通常用于數據傳輸,以及加密算法的基礎算法,傳輸過程中能夠保證數據傳輸的穩定性
BASE64是可以編碼和解碼的
*/
NSString *encodeStr = [receipt base64EncodedStringWithOptions:NSDataBase64EncodingEndLineWithLineFeed];
NSString *encodeLoad = [NSString stringWithFormat:@"{\"receipt-data\" : \"%@\"}", encodeStr];
NSData *receiptData = [encodeLoad dataUsingEncoding:NSUTF8StringEncoding];
// 發送網絡POST請求,對購買憑證進行驗證
// In the test environment, use https://sandbox.itunes.apple.com/verifyReceipt
// In the real environment, use https://buy.itunes.apple.com/verifyReceipt
NSString *serverStr = @"https://buy.itunes.apple.com/verifyReceipt";
if (flag) {
serverStr = @"https://sandbox.itunes.apple.com/verifyReceipt";
}
NSURL *storeURL = [NSURL URLWithString:serverStr];
NSMutableURLRequest *storeRequest = [NSMutableURLRequest requestWithURL:storeURL];
[storeRequest setHTTPMethod:@"POST"];
[storeRequest setHTTPBody:receiptData];
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
[NSURLConnection sendAsynchronousRequest:storeRequest
queue:queue
completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
if (connectionError) {
// 無法連接服務器,購買校驗失敗
// KIAPPurchVerFailed
} else {
NSError *error;
NSDictionary *jsonResponse = [NSJSONSerialization JSONObjectWithData:data
options:0
error:&error];
NSLog(@"----驗證結果 %@",jsonResponse);
if (!jsonResponse) {
// 蘋果服務器校驗數據返回為空校驗失敗
// KIAPPurchVerFailed
}
// 先驗證正式服務器,如果正式服務器返回21007再去蘋果測試服務器驗證,沙盒測試環境蘋果用的是測試服務器
NSString *status = [NSString stringWithFormat:@"%@",jsonResponse[@"status"]];
if (status && [status isEqualToString:@"21007"]) {
#warning - Todo 給ViewController提示
[self verifyPruchaseWithPaymentTransaction:transaction isTestServer:YES];
}else if(status && [status isEqualToString:@"0"]){
#warning - Todo 給ViewController提示
// kIAPPurchVerSuccwss 購買成功切驗證成功
}
}
}];
// 驗證成功與否都注銷交易,否則會出現虛假憑證信息一直驗證不通過,每次進程序都得輸入蘋果賬號
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
}
PS:驗證流程
訂單正確性的驗證:
1.iOS客戶端(購買成功)→ 到蘋果服務器驗證 → 蘋果服務器返回驗證結果,做相應處理
2.iOS客戶端(購買成功)→ 后臺 → 蘋果服務器驗證 → 蘋果服務器返回驗證結果,做相應處理
服務器要做的是:
1.接收iOS前端發過來的購買憑證。
2.判斷憑證是否已經存在或驗證過,然后存儲該憑證。
3.將該憑證發送到對應環境下的蘋果服務器驗證,并將驗證結果返回給客戶端。
4.根據需求,是否修改用戶相應信息。
注意事項
1.bundleID要與iTunes Connect上你App的相同,不然是請求不到產品信息的
2.在沙盒環境進行測試內購的時候,要使用沒有越獄的蘋果手機。
3.在沙盒環境下真機測試內購時,請去app store中注銷你的apple ID,不然發起支付購買請求后會直接case:SKPaymentTransactionStateFailed。使用沙盒測試員的賬號時不需要真正花錢的。
4.如果只添加了一個沙盒測試員賬號,當一個真機已經使用了這個賬號,另一個真機再使用這個賬號支付也是會發生錯誤的。那就去多建幾個沙盒測試員賬號使用不同的,反正也是免費的,填寫也很快。
5.監聽購買結果,當失敗和成功時代碼中要調用:
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
該方法通知蘋果支付隊列該交易已完成,不然就會已發起相同 ID 的商品購買就會有此項目將免費恢復的提示。
請在本地做一下憑證存儲
現在訂單正確性的驗證是:iOS客戶端(購買成功)→ 后臺→后臺到蘋果服務器驗證→處理后臺返回結果做相應邏輯處理。
注意:
如果Your App 和 Your Server 中斷了鏈接
當我們前端購買成功后,憑證(receipt)本地保留一份,當與后臺驗證成功后,再將本地保留的憑證刪除。
否者一直使用本地已經保留的憑證與后臺交互。
測試前提條件:
1.在itunesConnect中填寫測試賬號。
2.在itunesConnect中填寫稅務單(就是銀行賬號,開戶名,收款機構等等的稅務單)。
3.交易收據內容(receipt)
"receipt":
{
"original_purchase_date_pst":"2015-06-22 20:56:34 America/Los_Angeles", //購買時間,太平洋標準時間
"purchase_date_ms":"1435031794826", //購買時間毫秒
"unique_identifier":"5bcc5503dbcc886d10d09bef079dc9ab08ac11bb",//唯一標識符
"original_transaction_id":"1000000160390314", //原始交易ID
"bvrs":"1.0",//iPhone程序的版本號
"transaction_id":"1000000160390314", //交易的標識
"quantity":"1", //購買商品的數量
"unique_vendor_identifier":"AEEC55C0-FA41-426A-B9FC-324128342652", //開發商交易ID
"item_id":"1008526677",//App Store用來標識程序的字符串
"product_id":"cosmosbox.strikehero.gems60",//商品的標識
"purchase_date":"2015-06-23 03:56:34 Etc/GMT",//購買時間
"original_purchase_date":"2015-06-23 03:56:34 Etc/GMT", //原始購買時間
"purchase_date_pst":"2015-06-22 20:56:34 America/Los_Angeles",//太平洋標準時間
"bid":"com.cosmosbox.StrikeHero",//iPhone程序的bundle標識
"original_purchase_date_ms":"1435031794826"http://毫秒
}