談談蘋果應用內支付(IAP)的坑

前言

IAP支付的坑太多,這里寫一些高級點的坑。


一、請求商品

下面是請求商品的代碼:

- (void)validateProductIdentifier:(NSArray *)productIdentifier {
    SKProductsRequest *productRequest = [[SKProductsRequest alloc] initWithProductIdentifiers:[NSSet setWithArray:productIdentifier]];
    self.request = productRequest;
    productRequest.delegate = self;
    [productRequest start];
}

#pragma mark - SKProductsRequestDelegate Protocol
- (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response {
    
    self.products = response.products;
    [response.products enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        SKProduct *product = (SKProduct *)obj;
        NSLog(@"valid identifier: %@", product.productIdentifier);   
    }];
    
    for (NSString *invalidIdentifier in response.invalidProductIdentifiers) {
        // invalid identifier
        NSLog(@"invalid identifier: %@", invalidIdentifier);
    }
    
    if (![SKPaymentQueue canMakePayments]) {
        // display error UI ...
    }
    // display store UI ...
}

向蘋果服務器請求商品信息,是為了展示商店UI。請求到的SKProduct,包含了商品的標題、描述、價格、貨幣符號等信息。在國內,一般都是服務器接口提供商品信息,客戶端直接展示商店UI,用戶點擊購買的時候,才發起支付。所以,這種情況下,沒必要向蘋果服務器請求商品信息。因為請求商品信息時,蘋果服務器在海外,國內延遲大,慢的約六七秒,甚至有可能跳不到SKProductsRequest的代理方法里面,造成支付失敗。

解決辦法:

直接省略掉SKProductsRequest這個請求的創建發起。支付時,使用paymentWithProductIdentifier來直接生成SKPayment。

SKPayment *payment = [SKPayment paymentWithProductIdentifier:productIdentifier];
[[SKPaymentQueue defaultQueue] addPayment:payment];

二、receipt驗證

獲取receipt
    NSData *receipt;
    if (IOS7_OR_LATER) {
    // iOS 7 style app receipts
        NSURL *receiptURL = [[NSBundle mainBundle] appStoreReceiptURL];
        receipt = [NSData dataWithContentsOfURL:receiptURL];
    }else {
    // iOS 6 style transaction receipt
        receipt = transaction.transactionReceipt;
    }
驗證receipt

Receipt Validation Programming Guide
上面地址是receipt的驗證方法。出于安全考慮,app receipt需要第三方服務器來和蘋果服務器進行驗證。驗證后返回值是json字典。

關于字段status的說明如下:

For iOS 6 style transaction receipts, the status code reflects the status of the specific transaction’s receipt.

For iOS 7 style app receipts, the status code is reflects the status of the app receipt as a whole. For example, if you send a valid app receipt that contains an expired subscription, the response is 0 because the receipt as a whole is valid.

可以看到,iOS 6風格的receipt,包含的就是該筆transaction的receipt。

而iOS 7風格的receipt,包含的信息是一個列表,里面包含了很多transaction的信息,如果返回status=0,那么只是表示整個App的receipt驗證通過。

app端需要發receipt給服務端,服務端向蘋果服務器驗證receipt,然后返回status。注意!iOS 7風格的receipt包含了整個應用的所有的交易憑據,所以,status=0時,應該分發該receipt中所有transaction的商品。蘋果的驗證結果只告訴我們receipt有效還是無效,并不知道哪些transaction分發過商品,所以,服務端需要根據從數據庫里面查詢,排重,記錄,還要驗證該筆transaction是否為退過款的訂單,避免重復分發商品。

誤區:

使用[[NSBundle mainBundle] appStoreReceiptURL]獲得receipt,服務端卻試圖尋找最后一筆transaction信息。

正確姿勢:

應該分發該receipt中所有transaction的商品(重復使用的、退款的除外)

receipt JSON返回結果

iOS 7風格

{
    environment = Sandbox;
    receipt =     {
        "adam_id" = 0;
        "app_item_id" = 0;
        "application_version" = "1.0";
        "bundle_id" = "com.dianzhong.kuaikan";
        "download_id" = 0;
        "in_app" =         (
                        {
                "is_trial_period" = false;
                "original_purchase_date" = "2016-09-18 07:55:33 Etc/GMT";
                "original_purchase_date_ms" = 1474185333000;
                "original_purchase_date_pst" = "2016-09-18 00:55:33 America/Los_Angeles";
                "original_transaction_id" = 1000000236789335;
                "product_id" = "com.dianzhong.kuaikan6";
                "purchase_date" = "2016-09-18 07:55:33 Etc/GMT";
                "purchase_date_ms" = 1474185333000;
                "purchase_date_pst" = "2016-09-18 00:55:33 America/Los_Angeles";
                quantity = 1;
                "transaction_id" = 1000000236789335;
            },
            ...
        );
        "original_application_version" = "1.0";
        "original_purchase_date" = "2013-08-01 07:00:00 Etc/GMT";
        "original_purchase_date_ms" = 1375340400000;
        "original_purchase_date_pst" = "2013-08-01 00:00:00 America/Los_Angeles";
        "receipt_creation_date" = "2017-04-05 08:53:06 Etc/GMT";
        "receipt_creation_date_ms" = 1491382386000;
        "receipt_creation_date_pst" = "2017-04-05 01:53:06 America/Los_Angeles";
        "receipt_type" = ProductionSandbox;
        "request_date" = "2017-04-05 08:54:44 Etc/GMT";
        "request_date_ms" = 1491382484980;
        "request_date_pst" = "2017-04-05 01:54:44 America/Los_Angeles";
        "version_external_identifier" = 0;
    };
    status = 0;
}

iOS 6風格

{
    receipt =     {
        bid = "com.dianzhong.kuaikan";
        bvrs = "1.0";
        "item_id" = 1140823223;
        "original_purchase_date" = "2017-04-01 08:48:59 Etc/GMT";
        "original_purchase_date_ms" = 1491036539000;
        "original_purchase_date_pst" = "2017-04-01 01:48:59 America/Los_Angeles";
        "original_transaction_id" = 1000000286821320;
        "product_id" = "com.dianzhong.kuaikan12";
        "purchase_date" = "2017-04-01 08:48:59 Etc/GMT";
        "purchase_date_ms" = 1491036539000;
        "purchase_date_pst" = "2017-04-01 01:48:59 America/Los_Angeles";
        quantity = 1;
        "transaction_id" = 1000000286821320;
        "unique_identifier" = 367c781771909890ea8d59b25db3daf05ef0fbcb;
        "unique_vendor_identifier" = "7F144627-A82D-4D71-AACD-C3BAF2ED6684";
    };
    status = 0;
}

可以發現,兩者的結構基本一致,都包含一個名為receipt的字典,不同的是,iOS 7風格的receipt,把每一筆交易信息放到了in_app數組里。

服務端要做的事情:

當status=0時,記錄receipt中的全部交易信息。
服務端可根據transaction_id來記錄每一筆交易的信息,作為一條記錄,寫入數據庫,方便后續查詢和排重。

字段 類型 描述
transaction_id integer 交易號
original_transaction_id integer 原始交易號
product_id string 商品標識符
quantity integer 數量
purchase_date string 購買日期
original_purchase_date string 原始購買日期
purchase_date_ms integer 購買日期(ms)
original_purchase_date_ms integer 原始購買日期(ms)
purchase_date_pst string 購買日期(pst)
original_purchase_date_pst string 原始購買日期(pst)
cancellation_date string 取消購買的日期

流程如下:

服務端處理receipt的流程
退款的訂單

用戶退款過的訂單依然會在receipt中出現,因此App服務器實現驗證的時候需要能夠識別出已經被退款的訂單,不至于給退款的訂單發貨。

被退款訂單的唯一標識是:它帶有一個cancellation_date字段。

服務端驗證憑據時,如果有這個字段,則不分發商品。


三、receipt的安全

唐巧在他的《iOS應用內付費(IAP)開發步驟列表》中提到:

考慮到網絡異常情況,iOS端的發送憑證操作應該進行持久化,如果程序退出,崩潰或網絡異常,可以恢復重試。

實際上,我們不需要手動造車輪!

SKPaymentQueue只要被監聽,系統會遍歷該應用所有的transaction,只要沒有用finish?Transaction:?方法結束掉的transaction,都會重新出現在updatedTransactions:方法里。系統為我們做好了非常安全的存儲transaction和receipt的操作,底層原理目前還不清楚。本人試過,刪除應用后重新安裝,只要Bundle ID不變,依然能跳到updatedTransactions:方法里。

注意!如果在付款給蘋果之前,你們的后臺自己搞了個orderNum這樣一個訂單號出來,就要自己存儲這個orderNum了。
另外,既然后臺搞了個訂單號出來,就默認這個訂單號關聯了某一種商品,所以獲取receipt時就要用transaction.transactionReceipt,這樣才能保證獲取的receipt只包含1條交易信息。


關于receipt驗證的更多資料,請點擊這里

歡迎補充!本人qq:2224048633

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,622評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,716評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,746評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,991評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,706評論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,036評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,029評論 3 450
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,203評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,725評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,451評論 3 361
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,677評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,161評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,857評論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,266評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,606評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,407評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,643評論 2 380

推薦閱讀更多精彩內容