IOS开发中IAP的代码详解,反正看完这篇对IAP的了解就差不多了

最近开发一个项目,需要负责Apple Pay相关的业务,所以写这一篇文章来学习和研究Apple Pay相关的一些内容,那么,首先是这篇文章的目录:

目录

1.Apple Pay是什么?
2.Apple Pay和IAP的区别
3.IAP流程
4.如何在项目中接入Apple Pay?
5.Apple Pay的支付流程
6.Apple Pay内部的业务逻辑(通用)


Apple Pay是什么?

在Apple Pay的发布会上,Eddy Cue表示,苹果并没有兴趣建立一个收集用户数据的业务,苹果并不知道你购买了什么,不知道你是从哪里购买的,为了这个商品花了多少钱。

  • 所以这也就是苹果和支付宝,微信等最大的不同:Apple Pay并不会将资金存放在Apple Pay中。
  • Apple Pay其实也就是相当于一个卡包,替你保存银行卡的信息,只不过是将这个卡包虚拟化了而已,而且Apple Pay中存储的银行卡信息等都进行了加密,所以非常的安全(当然所有的安全都不会是绝对安全)
  • Apple Pay中的Pay业务并不是Apple自己的业务,Apple Pay本身只是一个第三方的桥梁:连接用户和银行。Pay业务只是银行和Apple之间所合作的一个业务,它和银行之间是强关联的关系,和Apple之间是弱关联的关系,没有银行也就没有pay了,但是支付宝和微信就不一样,用户将资金放在支付宝和微信中,即便没有银行,也可以直接进行支付。

Apple Pay和IAP的区别

什么是IAP呢?其实IAP就是In App purchase,即应用内购买,也就是江湖人称的内购,IAP(应用内购买)是最常用的一种支付方式,属于免费应用+应用内购买的模式。IAP主要是应用(App)和App Store服务器之间进行信息的传递,用户在APP内部进行内购操作相当于用户购买了App Store中的某个商品,这是用户和APP Store之间的交易,然后苹果从交易中抽取30%,APP的所有者获得70%。

Apple Pay则不然,Apple Pay实质上就是等同于用户使用银行卡进行刷卡消费,Apple Pay就是一个卡包的作用,它建立的是用户,银行,商家之间的关系:用户购买商家的商品进行消费的时候,实则是通过Apple Pay向银行发送付款信息,然后银行接受消息,进行付款交易,交易完成,用户获得商品,商家获得money。

理解Apple Pay和应用内支付之间的区别是非常重要的。Apple Pay用于销售物理商品,比如食品杂货、衣服和电器,也能用于支付俱乐部的会员资格、酒店预订以及演出门票。另一方面,应用内支付(IPA)只用于销售虚拟物品,如你的App里的高级内容,以及订阅数字内容。
PassKit框架为Apple Pay提供API,应用内支付(IAP)的API则由StoreKit框架提供。


IAP流程

什么是IAP?全称即In App Purchase,也就是我们所讲的苹果内购,IAP的流程分为两种,一种是直接使用Apple的服务器进行购买和验证另一种就是自己架设服务器进行验证。由于国内网络连接Apple服务器验证非常慢,而且也是为了防止黑客伪造购买凭证,通常都是选择自己架设服务器进行验证,那么在了解IAP之前,首先就应该要了解一些在IAP之中的理论词语。

  • Store Kit
    Store Kit代表App和App Store之间进行通信。程序将会从App Store接收那些你想要提供的产品的信息,并将它们显示出来供用户购买。
    当用户需要购买某件产品时,程序调用Store Kit来收集购买信息。

  • Products
    产品可以是任意一项你想要出售的特性。产品在iTunes Connect中被组织,这和你添加一个新的App是一样的。支持的产品种类共有四种:
    1.内容型。包括电子书,电子杂志,照片,插图,游戏关卡,游戏角色,和其他的数字内容。
    2.扩展功能。这些功能已经包含在App内部。在未购买之前被锁定。例如,你可以在一个游戏程序中包含若干个小游戏,用户可以分别来购买这些游戏。
    3.服务。允许程序对单次服务收费。比如录音服务。
    4.订阅。支持对内容或服务的扩展访问。例如,你的程序可以每周提供财务信息或游戏门户网站的信息。应该设定一个合理的更新周期,以避免过于频繁的
    提示困扰用户。

要记住:你将负责跟踪订阅的过期信息,并且管理续费。App Store不会替你监视订阅的周期,也不提供自动收费的机制>

  • 通过App Store注册产品
    每个你想要出售的产品都必须先要通过iTunes Connect在App Store注册。你需要提供产品的名称,描述和其他在程序中用到的元数据。而且需要为产品指定唯一的标识符。当你的程序利用Store Kit和App Store通信时,会使用产品标识来取回产品的信息。如果用户购买某个商品时,程序可以用该标识来将产品标注为“已购买”。
  • App Store产品种类简化为以下三种
    1.消耗性商品。 该类商品在需要时被单次购买。比如,单次服务。
    2.非消耗性商品。 该类商品只需被某个用户购买一次,一旦被购买,和该用户iTunes 账户关联的设备都可以使用此商品。Store Kit为在多个设备上重新存储非消耗性商品提供了内置的支持。
    3.订阅类。订阅类商品拥有以上两种类型的特性。和消耗性商品一样,订阅类商品可以被多次购买; 你可以在程序内部加入自己的订阅计划更新机制。 另外,订阅类商品必须提供给和某一用户关联的所有设备。In App Purchase期望订阅类商品可以通过外部服务器交付。你必须为多个设备的订阅服务提供相应的支持。

IAP流程之使用Apple服务器

这种模型,需要交付的产品已经在程序内部了。这种方式通常用在一些被锁定的功能上。也可以用来交付在程序束(App Bundle)中的内容。该方式的一个重要的优点是你可以及时的给客户交付产品,大多数的内置产品应该为非消耗商品。
注意:In App Purchase不提供购买补丁的功能。 如果需要更改app的bundle,你必须向App Store提交新的app版本。
为了标识产品,程序要在bundle中存储产品的标识符。内置模式下,Apple建议使用plist来纪录产品的标识符。 内容类应用可以使用折衷方式很方便的添加新的内容,而不改动程序本身。
当成功购买产品后,程序应将锁定的功能解锁,提供给用户。 解锁的最简单方式是修改程序偏好设置(Application Preferences)。 当用户备份手机数据的时候,程序偏好设置也会随之备份。 程序可能需要建议用户在购买产品后备份手机以免丢失购买的内容。

Apple服务器
  1. 程序通过bundle存储的plist文件得到产品标识符的列表。
  2. 程序将得到的产品ID向App Store发送请求,确认产品的信息。
  3. App Store返回产品信息。
  4. 程序把返回的产品信息显示给用户(App的store界面)
  5. 用户选择某个产品
  6. 程序向App Store发送支付请求
  7. App Store处理支付请求并返回交易完成信息。
  8. App获取信息并提供内容给用户。

程序过程

  • IAP开发前的准备

** 第一步:创建APP内购项目**
iTunesConnect是苹果提供的一个平台,主要提供APP发布和管理App的,最重要的功能是创建管理项目信息,项目付费产品(道具)管理、付费的测试账号、提交App等等。
首先需要登录iTunesConnect,在平台上填写APP的内购产品的相关信息

注意:产品的buddle ID是不可变的,但是产品ID等信息后期是可以修改的,这里的Bundle ID一定要跟项目中的info.plist中的Bundle ID保证一致,一般的buddle ID 的格式如下:com.domainname.appname

(bundle ID可以翻译成包ID,也可以叫APP ID 或应用ID,它是每一个ios应用的全球唯一标识。无论代码怎么改,图标和应用名称怎么换,只要bundle id没变,ios系统就认为这是同一个应用。每开发一个新应用,首先都需要到member center->identifier->APP IDS去创建一个bundle id)
(ios certificates就是证书。它的作用就是证明你的mac具有开发或发布某个开发者账号下应用的权限。而且证书还分成两种,一种是开发证书,也叫Development certificate; 另一种是发布证书或叫生产证书,英文名叫Production certificate)

第二步:添加沙盒测试账号
用户和职能中选择沙箱技术测试人员,然后添加上技术测试账号,测试账号可以填的较为随意。

  • IAP的流程图
屏幕快照 2017-10-09 下午9.14.01.png
  • IAP开发的代码过程
    1.在工程中引入storekit.framework并且import <StoreKit/StoreKit.h>
    2.获得所有产品的Product ID,并且用常量存储在本地,产品ID也可以由自己的服务器返回。
    3.制作一个界面,展示所有的应用内付费项目。这些应用内付费项目的价格和介绍信息可以是自己的服务器返回。但如果是不带服务器的单机游戏应用或工具类应用, 则可以通过向App Store查询获得。实际上向App Store查询速度非常慢,通常需要2-3秒钟,所以不建议这么做,最好还是搞个自己的服务器进行操作。
    4.用户点击一个IAP项目的时候,首先询问用户是否允许应用购买
//点击购买按钮
  - (void)clickPurcaseBtnAction
  {
      //点击按钮的时候判断app是否允许apple支付
      if ([SKPaymentQueue canMakePayments]) {
          //请求苹果后台商品
          [self requestProductData:_productID];
      }
      else
      {
          NSLog(@"用户不允许应用内购买");
      }
 }

5.通过自己获得的产品ID向Apple store发出请求,产品ID确认成功的时候创建SKPayment实例,发起购买操作

    //去苹果服务器请求商品
 -(void)requestProductData:(NSString *)type {
         //根据商品ID查找商品信息
          NSArray *product = [[NSArray alloc] initWithObjects:type, nil];
          NSSet *nsset = [NSSet setWithArray:product];
          //创建SKProductsRequest对象,用想要出售的商品的标识来初始化,然后附加上对应的委托对象。              
          //该请求的响应包含了可用商品的本地化信息。
          SKProductsRequest *request = [[SKProductsRequest alloc]  initWithProductIdentifiers:nsset];
          request.delegate = self;
          [request start];
}

6.查询请求结束后会有相对应的回调(查询成功或者查询失败)

      // SKProductsRequestDelegate
      //查询成功的回调
      -(void)productsRequest:(SKProductsRequest *)request didReceiveResponse:  (SKProductsResponse *)response {
         NSArray *product = response.products;
      //invalidProductIdentifiers是不被App Store所识别的产品id字符串数组,通常为空
         NSLog(@"产品Product ID:%@",response.invalidProductIdentifiers);
         if(product == nil) {
           return;
         }
      //数组的count代表回调的产品ID数组的长度
         if (product.count == 0) {
            NSLog(@"无法获得产品信息,购买失败");
            return;
          }
      //SKProduct对象包含了在App Store上注册的商品的本地化信息。
      SKProduct *storeProduct = nil;
       for (SKProduct *pro in product) {
          if ([pro.productIdentifier isEqualToString:_productID]) {
              storeProduct = pro;
          }
      }
      if(storeProduct == nil) {
              return;
      }
      //创建一个支付对象,并放到队列中
      self.g_payment = [SKMutablePayment paymentWithProduct:storeProduct];
      //设置购买的数量
      self.g_payment.quantity = 1;
      [[SKPaymentQueue defaultQueue] addPayment:self.g_payment];
      }


    //查询失败的回调
    - (void)request:(SKRequest *)request didFailWithError:(NSError *)error {
        CTHLog(@"请求商品失败%@", error);
    }

7.用户的交易有结果是会调用下方的交易回调函数

//交易有结果时会调用此回调函数,SKPaymentTransactionObserver - 交易的回调
- (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray<SKPaymentTransaction *> *)transactions {
    for (SKPaymentTransaction *transaction in transactions) {
        switch (transaction.transactionState) {
            case SKPaymentTransactionStatePurchasing: //正在购买
                [self purchasingTransaction:transaction];
                break;
            case SKPaymentTransactionStatePurchased:  //购买成功
                [self completedTransaction:transaction Product:self.myProduct];
                break;
            case SKPaymentTransactionStateFailed:     //购买失败
                [self failedTransaction:transaction];
                break;
            case SKPaymentTransactionStateRestored:   //恢复购买
                [self restoredTransaction:transaction];
                break;
            case SKPaymentTransactionStateDeferred:   //最终状态未确认
                [self deferredTransaction:transaction];
                break;
            default:
                break;
        }
    }
}

8.苹果服务器验证凭据

- (void)verifyReceipt {
    //1.appStoreReceiptURL ios 7.0增加的,购买交易完成之后,会将凭据存在本地沙盒
    NSURL *receiptLocal = [[NSBundle mainBundle] appStoreReceiptURL];
    NSData *receiptData = [NSData dataWithContentsOfURL:receiptLocal];
    
    //2.本地凭据的解码(base64解码)
    NSString *encodeStr = [receiptData base64EncodedStringWithOptions:NSDataBase64EncodingEndLineWithLineFeed];
    NSString *payload = [NSString stringWithFormat:@"{\"receipt-data\" : \"%@\", \"password\" : \"aaaaaaaaa\"}", encodeStr];
    NSData *payloadData = [payload dataUsingEncoding:NSUTF8StringEncoding];
    
    //2.组合要想服务器发送的请求:HTTPMethod,HTTPBody
    NSURL *verifyUrl = [NSURL URLWithString:buyUrl];
    
    NSMutableURLRequest *urlRequest = [NSMutableURLRequest requestWithURL:verifyUrl cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:10.0f];
    urlRequest.HTTPMethod = @"POST";
    urlRequest.HTTPBody = payloadData;
    
    //3.验证结果
    NSURLSession *session = [NSURLSession sharedSession];
    NSURLSessionDataTask *sessionDataTask = [session dataTaskWithRequest:urlRequest completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
        // 官方验证结果为空
        if (data == nil)
        {
            CTHLog(@"验证失败");
            return;
        }
        NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingAllowFragments error:nil];
        if (dict != nil) {
            //~验证成功
            if ([dict[@"status"] intValue] == 0) {
             
            }
            else {
                CTHLog(@"验证失败");
            }
            
        }
        else {
            CTHLog(@"验证失败");
        }
        
    }];
    [sessionDataTask resume];
}

苹果测试的地址为: https://sandbox.itunes.apple.com/verifyReceipt
苹果正式购买的地址为:https://buy.itunes.apple.com/verifyReceipt


IAP流程之自己架设服务器

使用这种方式,要提供另外的服务器将产品发送给程序。 服务器交付适用于订阅、内容类商品和服务,因为商品可以作为数据发送,而不需改动程序束。 例如,一个游戏提供的新的内容(关卡等)。 Store Kit不会对服务器端的设计和交互做出定义,这方面工作需要你来完成。 而且,Store Kit不提供验证用户身份的机制,你需要来设计。 如果你的程序需要以上功能,例如,纪录特定用户的订阅计划, 你需要自己来设计和实现。

Apple建议在服务器端存储产品标识,而不是将其存储在plist中。这样就可以在不升级程序的前提下添加新的产品。

在服务器模式下,你的程序将获得交易(transaction)相关的信息,并将它发送给服务器。服务器可以验证收到的数据,并将其解码已确定需要交付的内容。


自己架设服务器
  1. 程序向服务器发送请求,获得一份产品列表。
  2. 服务器返回包含产品标识符的列表。
  3. 程序向App Store发送请求,得到产品的信息。
  4. App Store返回产品信息。
  5. 程序把返回的产品信息显示给用户(App的store界面)
  6. 用户选择某个产品
  7. 程序向App Store发送支付请求
  8. App Store处理支付请求并返回交易完成信息。
  9. 程序从信息中获得数据,并发送至服务器。
  10. 服务器纪录数据,并进行审(我们的)查。
  11. 服务器将数据发给App Store来验证该交易的有效性。
  12. App Store对收到的数据进行解析,返回该数据和说明其是否有效的标识。
  13. 服务器读取返回的数据,确定用户购买的内容。
  14. 服务器将购买的内容传递给程序。

接下来对这第二条架设服务器这14条进行一个总结(上文所述已有总结)

  • 用户进入购买虚拟物品页面,App从后台服务器获取产品列表然后显示给用户
  • 用户点击购买购买某一个虚拟物品,APP就发送该虚拟物品的productionIdentifier到Apple服务器
  • Apple服务器根据APP发送过来的productionIdentifier返回相应的物品的信息(描述,价格等)
  • 用户点击确认键购买该物品,购买请求发送到Apple服务器
  • Apple服务器完成购买后,返回用户一个完成购买的凭证
  • APP发送这个凭证到后台服务器验证
  • 后台服务器把这个凭证发送到Apple验证,Apple返回一个字段给后台服务器表明该凭证是否有效
  • 后台服务器把验证结果在发送到APP,APP根据验证结果做相应的处理

如何在项目中接入Apple Pay?

为了使Apple Pay生效,除了PassKit框架之外,还需要:

  1. 建立一个拥有支付模块或通道的账户(如果你没有的话)
  2. Certificates, Identifiers & Profiles注册一个商业标示符
  3. 提交一个证书签名需求以获得用于加密和解码支付令牌的公开或私有密钥
  4. 在你的App里包含Apple Pay的支持权限

以下是较为具体的代码代码流程:
Apple Pay接入详细教程
Apple Pay接入文档

Apple Pay的支付流程

Apple并不处理和付款相关的逻辑,它只是负责支付信息的传递。Apple通过Touch ID来验证银行卡持有者的身份,实际的扣款行为发生在银联端,接入了Apple Pay的商品(即App)组织好Apple返回的支付信息,向银行发出扣款请求后,该笔交易才会真正发送扣款,所以APP本身是和银联进行结算,Apple Pay只不过是作为一种支付的渠道而已。

支付宝类
Apple Pay

Apple Pay内部的业务逻辑

业务逻辑

Apple Pay在应用内的支付流程如下:

  1. APP根据使用场景显示Payment Sheet
  2. 用户选择需要支付的卡以及支付需要的个人信息后,进行指纹验证,之后根据情况,有些银行卡还需要输入卡所对应的密码(PIN码)
  3. App将支付相关的信息发送到Apple的服务器,进行加密。然后通过回调函数将加密后的支付信息返回给相应的App
  4. APP在收到回调之后,将对应的信息发送到自己的服务器中
  5. 服务器在收到APP发送过来的支付信息后,将数据进行解密操作,提取其中需要的信息,组织银行接口报文,调用银行的接口,完成扣款。

注意点:

  1. App 收到的 Payment sheet 回调信息中,包含了一个 PKPayment 的对象,该对象包含了所有跟 Apple Pay 支付相关所有信息。比如用户的手机号或者收货地址等等,其中最重要的就是 payment token,它的 paymentData 字段数据就是需要发送给服务器的内容。用户信息部分是明文的,而支付信息也就是 paymentData 部分则是被加密过的。

  2. paymentData 的内容是 Json 格式的二进制流,服务器在收到这个数据之后进行解析,其中的 header.wrappedKey 是使用非对称加密算法加密过的对称秘钥。使用在苹果开发者后台配置 merchant 时的私钥进行解密,会得到这个对称秘钥。然后用这个对称秘钥对 data 字段所包含的加密数据进行解密,可以得到 Apple 返回的与支付相关的信息。此支付信息是加密过的,包含了用户支付的卡号和 PIN 码等信息,理论上只有银联才能解析出来真正的内容,我们作为商户是看不到具体信息的。服务器端需要将这些解密过的信息组织成银联所需的报文内容,然后调用银联的扣款接口,完成扣款。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • App Store 审核指南2017/07/27 简介 App 正在改变世界,丰富人们的生活,并为像您一样的开发者...
    小白沐春风阅读 3,567评论 0 2
  • 简介 App 正在改变世界,丰富人们的生活,并为像您一样的开发者提供前所未有的创新机会。因此,App Store ...
    o0_0o阅读 3,343评论 2 48
  • 昨天,在521这个日子里,很意外的大家没有开启虐狗模式,大家很和谐的聚在一起进行了我们两年以来第一次人全部到齐的宿...
    期迹阅读 1,910评论 0 1
  • 1.谈谈你在阅读这篇文章中的心理感受变化。 第一次阅读,开始并没有看进去,可能是不习惯这样的文字风格。但是随着一些...
    会飞的披萨店阅读 136评论 0 0
  • 安南寺虽然有风轩,但是他不认得我。寺外虽然有初寒,但是两个人干巴巴的说话太多,也未免心生烦躁。 于是我经常会回南海...
    冷清持阅读 664评论 3 15