Files
SwiftyStoreKit/README.md

625 lines
31 KiB
Markdown
Raw Normal View History

2016-10-20 21:09:02 +01:00
![](https://github.com/bizz84/SwiftyStoreKit/raw/master/SwiftyStoreKit-logo.png)
2015-09-03 18:58:51 +01:00
2016-09-15 22:33:27 +01:00
[![License](https://img.shields.io/badge/license-MIT-blue.svg?style=flat)](http://mit-license.org)
2016-09-16 07:31:31 +01:00
[![Platform](http://img.shields.io/badge/platform-ios%20%7C%20macos%20%7C%20tvos-lightgrey.svg?style=flat)](https://developer.apple.com/resources/)
[![Language](https://img.shields.io/badge/swift-3.0-orange.svg)](https://developer.apple.com/swift)
2016-09-15 22:33:27 +01:00
[![Build](https://img.shields.io/travis/bizz84/SwiftyStoreKit.svg?style=flat)](https://travis-ci.org/bizz84/SwiftyStoreKit)
[![Issues](https://img.shields.io/github/issues/bizz84/SwiftyStoreKit.svg?style=flat)](https://github.com/bizz84/SwiftyStoreKit/issues)
2015-12-26 01:29:28 +01:00
[![Cocoapod](http://img.shields.io/cocoapods/v/SwiftyStoreKit.svg?style=flat)](http://cocoadocs.org/docsets/SwiftyStoreKit/)
[![Carthage compatible](https://img.shields.io/badge/Carthage-compatible-4BC51D.svg?style=flat)](https://github.com/Carthage/Carthage)
2016-09-15 22:58:31 +01:00
[![Twitter](https://img.shields.io/badge/twitter-@biz84-blue.svg?maxAge=2592000)](http://twitter.com/biz84)
2016-11-05 00:00:12 +00:00
SwiftyStoreKit is a lightweight In App Purchases framework for iOS 8.0+, tvOS 9.0+ and macOS 10.10+.
2016-10-20 21:09:02 +01:00
### Preview
2017-12-29 07:44:04 +00:00
<img src="https://github.com/bizz84/SwiftyStoreKit/raw/master/Screenshots/Preview.jpg" width="320">
### Note from the Author
2017-09-05 13:43:34 +01:00
I started [**Sustainable Earth**](https://github.com/bizz84/Sustainable-Earth), a curated list of all things sustainable. Interested? [It's on GitHub](https://github.com/bizz84/Sustainable-Earth).
2017-04-24 18:33:22 +01:00
## Contributing
#### Got issues / pull requests / want to contribute? [Read here](CONTRIBUTING.md).
## App startup
### Complete Transactions
Apple recommends to register a transaction observer [as soon as the app starts](https://developer.apple.com/library/ios/technotes/tn2387/_index.html):
> Adding your app's observer at launch ensures that it will persist during all launches of your app, thus allowing your app to receive all the payment queue notifications.
SwiftyStoreKit supports this by calling `completeTransactions()` when the app starts:
```swift
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
2017-12-22 16:30:09 +00:00
// see notes below for the meaning of Atomic / Non-Atomic
SwiftyStoreKit.completeTransactions(atomically: true) { purchases in
for purchase in purchases {
switch purchase.transaction.transactionState {
case .purchased, .restored:
if purchase.needsFinishTransaction {
// Deliver content from server, then:
SwiftyStoreKit.finishTransaction(purchase.transaction)
}
// Unlock content
case .failed, .purchasing, .deferred:
break // do nothing
}
}
}
2017-07-06 14:34:14 +09:00
return true
}
```
If there are any pending transactions at this point, these will be reported by the completion block so that the app state and UI can be updated.
If there are no pending transactions, the completion block will **not** be called.
Note that `completeTransactions()` **should only be called once** in your code, in `application(:didFinishLaunchingWithOptions:)`.
## Purchases
### Retrieve products info
```swift
SwiftyStoreKit.retrieveProductsInfo(["com.musevisions.SwiftyStoreKit.Purchase1"]) { result in
if let product = result.retrievedProducts.first {
let priceString = product.localizedPrice!
print("Product: \(product.localizedDescription), price: \(priceString)")
}
else if let invalidProductId = result.invalidProductIDs.first {
print("Invalid product identifier: \(invalidProductId)")
}
else {
2017-12-28 22:00:48 +00:00
print("Error: \(result.error)")
}
}
```
### Purchase a product (given a product id)
2015-09-03 18:58:51 +01:00
* **Atomic**: to be used when the content is delivered immediately.
2015-09-03 18:58:51 +01:00
```swift
SwiftyStoreKit.purchaseProduct("com.musevisions.SwiftyStoreKit.Purchase1", quantity: 1, atomically: true) { result in
2015-09-03 18:58:51 +01:00
switch result {
case .success(let purchase):
print("Purchase Success: \(purchase.productId)")
case .error(let error):
2017-02-20 18:29:15 +00:00
switch error.code {
case .unknown: print("Unknown error. Please contact support")
case .clientInvalid: print("Not allowed to make the payment")
case .paymentCancelled: break
case .paymentInvalid: print("The purchase identifier was invalid")
case .paymentNotAllowed: print("The device is not allowed to make the payment")
case .storeProductNotAvailable: print("The product is not available in the current storefront")
case .cloudServicePermissionDenied: print("Access to cloud service information is not allowed")
case .cloudServiceNetworkConnectionFailed: print("Could not connect to the network")
case .cloudServiceRevoked: print("User has revoked permission to use this cloud service")
2017-02-20 18:29:15 +00:00
}
}
}
```
* **Non-Atomic**: to be used when the content is delivered by the server.
```swift
SwiftyStoreKit.purchaseProduct("com.musevisions.SwiftyStoreKit.Purchase1", quantity: 1, atomically: false) { result in
switch result {
case .success(let product):
// fetch content from your server, then:
if product.needsFinishTransaction {
SwiftyStoreKit.finishTransaction(product.transaction)
}
print("Purchase Success: \(product.productId)")
2016-07-22 09:48:30 +01:00
case .error(let error):
2017-02-20 18:29:15 +00:00
switch error.code {
case .unknown: print("Unknown error. Please contact support")
case .clientInvalid: print("Not allowed to make the payment")
case .paymentCancelled: break
case .paymentInvalid: print("The purchase identifier was invalid")
case .paymentNotAllowed: print("The device is not allowed to make the payment")
case .storeProductNotAvailable: print("The product is not available in the current storefront")
case .cloudServicePermissionDenied: print("Access to cloud service information is not allowed")
case .cloudServiceNetworkConnectionFailed: print("Could not connect to the network")
case .cloudServiceRevoked: print("User has revoked permission to use this cloud service")
2017-02-20 18:29:15 +00:00
}
2015-09-03 18:58:51 +01:00
}
}
```
### Purchase a product (given a SKProduct)
This is a variant of the method above that can be used to purchase a product when the corresponding `SKProduct` has already been retrieved with `retrieveProductsInfo`:
```swift
SwiftyStoreKit.retrieveProductsInfo(["com.musevisions.SwiftyStoreKit.Purchase1"]) { result in
if let product = result.retrievedProducts.first {
SwiftyStoreKit.purchaseProduct(product, quantity: 1, atomically: true) { result in
// handle result (same as above)
}
}
}
```
Using this `purchaseProduct` method guarantees that only one network call is made to StoreKit to perform the purchase, as opposed to one call to get the product and another to perform the purchase.
### Should add store payment handling (iOS 11)
iOS 11 adds a new delegate method on `SKPaymentTransactionObserver`:
```swift
@available(iOS 11.0, *)
optional public func paymentQueue(_ queue: SKPaymentQueue, shouldAddStorePayment payment: SKPayment, for product: SKProduct) -> Bool
```
From [Apple Docs](https://developer.apple.com/documentation/storekit/skpaymenttransactionobserver/2877502-paymentqueue):
> This delegate method is called when the user has started an in-app purchase in the App Store, and is continuing the transaction in your app. Specifically, if your app is already installed, the method is called automatically.
If your app is not yet installed when the user starts the in-app purchase in the App Store, the user gets a notification when the app installation is complete. This method is called when the user taps the notification. Otherwise, if the user opens the app manually, this method is called only if the app is opened soon after the purchase was started.
SwiftyStoreKit supports this with a new handler, called like this:
```swift
SwiftyStoreKit.shouldAddStorePaymentHandler = { payment, product in
2017-12-28 22:00:48 +00:00
// return true if the content can be delivered by your app
// return false otherwise
}
```
To test this in sandbox mode, open this URL in Safari:
```
itms-services://?action=purchaseIntent&bundleId=com.example.app&productIdentifier=product_name
```
More information on the [WWDC17 session What's New in StoreKit](https://developer.apple.com/videos/play/wwdc2017/303)
([slide number 165](https://devstreaming-cdn.apple.com/videos/wwdc/2017/303f0u5froddl13/303/303_whats_new_in_storekit.pdf) shows the link above).
2015-09-04 12:04:10 +01:00
### Restore previous purchases
2015-09-03 18:58:51 +01:00
According to [Apple - Restoring Purchased Products](https://developer.apple.com/library/content/documentation/NetworkingInternet/Conceptual/StoreKitGuide/Chapters/Restoring.html#//apple_ref/doc/uid/TP40008267-CH8-SW9):
> In most cases, all your app needs to do is refresh its receipt and deliver the products in its receipt. The refreshed receipt contains a record of the users purchases in this app, on this device or any other device.
> Restoring completed transactions creates a new transaction for every completed transaction the user made, essentially replaying history for your transaction queue observer.
See the **Receipt Verification** section below for how to restore previous purchases using the receipt.
This section shows how to restore completed transactions with the `restorePurchases` method instead. When successful, the method returns all non-consumable purchases, as well as all auto-renewable subscription purchases, **regardless of whether they are expired or not**.
* **Atomic**: to be used when the content is delivered immediately.
2015-09-03 18:58:51 +01:00
```swift
SwiftyStoreKit.restorePurchases(atomically: true) { results in
if results.restoreFailedPurchases.count > 0 {
print("Restore Failed: \(results.restoreFailedPurchases)")
}
else if results.restoredPurchases.count > 0 {
2017-06-13 15:13:06 +02:00
print("Restore Success: \(results.restoredPurchases)")
}
else {
print("Nothing to Restore")
}
}
```
* **Non-Atomic**: to be used when the content is delivered by the server.
```swift
SwiftyStoreKit.restorePurchases(atomically: false) { results in
if results.restoreFailedPurchases.count > 0 {
print("Restore Failed: \(results.restoreFailedPurchases)")
}
else if results.restoredPurchases.count > 0 {
for purchase in results.restoredPurchases {
// fetch content from your server, then:
if purchase.needsFinishTransaction {
SwiftyStoreKit.finishTransaction(purchase.transaction)
}
}
print("Restore Success: \(results.restoredPurchases)")
}
else {
2015-09-03 18:58:51 +01:00
print("Nothing to Restore")
}
}
```
2015-09-04 11:05:00 +01:00
#### What does atomic / non-atomic mean?
When you purchase a product the following things happen:
* A payment is added to the payment queue for your IAP.
* When the payment has been processed with Apple, the payment queue is updated so that the appropriate transaction can be handled.
* If the transaction state is **purchased** or **restored**, the app can unlock the functionality purchased by the user.
* The app should call `finishTransaction(_:)` to complete the purchase.
This is what is [recommended by Apple](https://developer.apple.com/reference/storekit/skpaymentqueue/1506003-finishtransaction):
> Your application should call `finishTransaction(_:)` only after it has successfully processed the transaction and unlocked the functionality purchased by the user.
* A purchase is **atomic** when the app unlocks the functionality purchased by the user immediately and call `finishTransaction(_:)` at the same time. This is desirable if you're unlocking functionality that is already inside the app.
* In cases when you need to make a request to your own server in order to unlock the functionality, you can use a **non-atomic** purchase instead.
* **Note**: SwiftyStoreKit doesn't yet support downloading content hosted by Apple for non-consumable products. See [this feature request](https://github.com/bizz84/SwiftyStoreKit/issues/128).
2016-11-16 22:32:32 +00:00
SwiftyStoreKit provides three operations that can be performed **atomically** or **non-atomically**:
* Making a purchase
* Restoring purchases
* Completing transactions on app launch
## Receipt verification
According to [Apple - Delivering Products](https://developer.apple.com/library/content/documentation/NetworkingInternet/Conceptual/StoreKitGuide/Chapters/DeliverProduct.html#//apple_ref/doc/uid/TP40008267-CH5-SW4):
> The app receipt contains a record of the users purchases, cryptographically signed by Apple. For more information, see [Receipt Validation Programming Guide](https://developer.apple.com/library/content/releasenotes/General/ValidateAppStoreReceipt/Introduction.html#//apple_ref/doc/uid/TP40010573).
> Information about consumable products is added to the receipt when theyre paid for and remains in the receipt until you finish the transaction. After you finish the transaction, this information is removed the next time the receipt is updated—for example, the next time the user makes a purchase.
> Information about all other kinds of purchases is added to the receipt when theyre paid for and remains in the receipt indefinitely.
2017-05-18 08:37:48 +02:00
When an app is first installed, the app receipt is missing.
2017-10-11 15:19:01 +02:00
As soon as a user completes a purchase or restores purchases, StoreKit creates and stores the receipt locally as a file, located by `Bundle.main.appStoreReceiptURL`.
2017-05-18 08:37:48 +02:00
2017-10-11 15:37:47 +02:00
### Retrieve local receipt (encrypted)
2017-10-11 15:19:01 +02:00
This helper can be used to retrieve the (encrypted) local receipt data:
```swift
let receiptData = SwiftyStoreKit.localReceiptData
let receiptString = receiptData.base64EncodedString(options: [])
// do your receipt validation here
```
2017-05-18 08:37:48 +02:00
2017-10-11 15:19:01 +02:00
However, the receipt file may be missing or outdated.
2017-05-18 08:37:48 +02:00
2017-10-11 15:37:47 +02:00
### Fetch receipt (encrypted)
2017-05-18 08:37:48 +02:00
2017-10-11 15:19:01 +02:00
Use this method to get the updated receipt:
```swift
SwiftyStoreKit.fetchReceipt(forceRefresh: true) { result in
switch result {
case .success(let receiptData):
let encryptedReceipt = receiptData.base64EncodedString(options: [])
2017-10-11 20:07:09 +02:00
print("Fetch receipt success:\n\(encryptedReceipt)")
2017-10-11 15:19:01 +02:00
case .error(let error):
2017-10-11 20:07:09 +02:00
print("Fetch receipt failed: \(error)")
2017-10-11 15:19:01 +02:00
}
}
```
This method works as follows:
* If `forceRefresh = false`, it returns the local receipt from file, or refreshes it if missing.
* If `forceRefresh = true`, it always refreshes the receipt regardless.
2017-05-18 08:37:48 +02:00
2017-05-18 10:47:49 +01:00
**Notes**
2017-10-11 15:19:01 +02:00
* If the local receipt is missing or `forceRefresh = true` when calling `fetchReceipt`, a network call is made to refresh it.
2017-05-18 10:47:49 +01:00
* If the user is not logged to the App Store, StoreKit will present a popup asking to **Sign In to the iTunes Store**.
2017-10-11 20:07:09 +02:00
* If the user enters valid credentials, the receipt will be refreshed.
2017-05-18 10:47:49 +01:00
* If the user cancels, receipt refresh will fail with a **Cannot connect to iTunes Store** error.
2017-10-11 15:19:01 +02:00
If `fetchReceipt` is successful, it will return the **encrypted** receipt as a string. For this reason, a **validation** step is needed to get all the receipt fields in readable form. This can be done in various ways:
2017-10-11 15:34:59 +02:00
1. Validate with Apple via the `AppleReceiptValidator` (see [`verifyReceipt`](#verify-receipt) below).
2017-10-11 15:19:01 +02:00
2. Perform local receipt validation (see [#101](https://github.com/bizz84/SwiftyStoreKit/issues/101)).
3. Post the receipt data and validate on server.
### Verify Receipt
2017-10-11 15:19:01 +02:00
Use this method to (optionally) refresh the receipt and perform validation in one step.
2016-03-30 22:47:01 +02:00
```swift
2017-10-11 17:42:33 +02:00
let appleValidator = AppleReceiptValidator(service: .production, sharedSecret: "your-shared-secret")
SwiftyStoreKit.verifyReceipt(using: appleValidator, forceRefresh: false) { result in
switch result {
case .success(let receipt):
2017-10-11 20:07:09 +02:00
print("Verify receipt success: \(receipt)")
case .error(let error):
2017-10-11 20:07:09 +02:00
print("Verify receipt failed: \(error)")
}
}
```
2017-10-11 15:19:01 +02:00
**Notes**
* This method is based on `fetchReceipt`, and the same refresh logic discussed above applies.
2017-10-11 15:34:59 +02:00
* `AppleReceiptValidator` is a **reference implementation** that validates the receipt with Apple and results in a network call. _This is prone to man-in-the-middle attacks._
2017-10-11 15:19:01 +02:00
* You should implement your secure logic by validating your receipt locally, or sending the encrypted receipt data and validating it in your server.
* Local receipt validation is not implemented (see [issue #101](https://github.com/bizz84/SwiftyStoreKit/issues/101) for details).
2017-10-11 15:34:59 +02:00
* You can implement your own receipt validator by conforming to the `ReceiptValidator` protocol and passing it to `verifyReceipt`.
2017-05-18 10:47:49 +01:00
## Verifying purchases and subscriptions
2017-05-18 10:47:49 +01:00
Once you have retrieved the receipt using the `verifyReceipt` method, you can verify your purchases and subscriptions by product identifier.
Verifying multiple purchases and subscriptions in one call is not yet supported (see [issue #194](https://github.com/bizz84/SwiftyStoreKit/issues/194) for more details).
2017-05-18 10:47:49 +01:00
If you need to verify multiple purchases / subscriptions, you can either:
* manually parse the receipt dictionary returned by `verifyReceipt`
* call `verifyPurchase` or `verifySubscription` multiple times with different product identifiers
### Verify Purchase
2016-05-18 18:31:17 +07:00
```swift
2017-10-11 17:42:33 +02:00
let appleValidator = AppleReceiptValidator(service: .production, sharedSecret: "your-shared-secret")
SwiftyStoreKit.verifyReceipt(using: appleValidator) { result in
2016-05-18 18:31:17 +07:00
switch result {
2016-07-22 09:48:30 +01:00
case .success(let receipt):
2016-05-24 11:57:31 +07:00
// Verify the purchase of Consumable or NonConsumable
2016-05-18 18:31:17 +07:00
let purchaseResult = SwiftyStoreKit.verifyPurchase(
productId: "com.musevisions.SwiftyStoreKit.Purchase1",
inReceipt: receipt)
2016-05-18 18:31:17 +07:00
switch purchaseResult {
case .purchased(let receiptItem):
print("Product is purchased: \(receiptItem)")
2016-07-22 09:48:30 +01:00
case .notPurchased:
2016-05-23 18:46:23 +07:00
print("The user has never purchased this product")
}
2017-02-20 18:29:15 +00:00
case .error(let error):
print("Receipt verification failed: \(error)")
}
}
```
Note that for consumable products, the receipt will only include the information for a couple of minutes after the purchase.
### Verify Subscription
This can be used to check if a subscription was previously purchased, and whether it is still active or if it's expired.
From [Apple - Working with Subscriptions](https://developer.apple.com/library/content/documentation/NetworkingInternet/Conceptual/StoreKitGuide/Chapters/Subscriptions.html#//apple_ref/doc/uid/TP40008267-CH7-SW6):
> keep a record of the date that each piece of content is published. Read the Original Purchase Date and Subscription Expiration Date field from each receipt entry to determine the start and end dates of the subscription.
When one or more subscriptions are found for a given product id, they are returned as a `ReceiptItem` array ordered by `expiryDate`, with the first one being the newest.
```swift
2017-10-11 17:42:33 +02:00
let appleValidator = AppleReceiptValidator(service: .production, sharedSecret: "your-shared-secret")
SwiftyStoreKit.verifyReceipt(using: appleValidator) { result in
switch result {
2016-07-22 09:48:30 +01:00
case .success(let receipt):
// Verify the purchase of a Subscription
2016-05-24 11:57:31 +07:00
let purchaseResult = SwiftyStoreKit.verifySubscription(
type: .autoRenewable, // or .nonRenewing (see below)
productId: "com.musevisions.SwiftyStoreKit.Subscription",
inReceipt: receipt)
2016-05-24 11:57:31 +07:00
switch purchaseResult {
case .purchased(let expiryDate, let receiptItems):
print("Product is valid until \(expiryDate)")
case .expired(let expiryDate, let receiptItems):
print("Product is expired since \(expiryDate)")
2016-07-22 09:48:30 +01:00
case .notPurchased:
2016-05-24 11:57:31 +07:00
print("The user has never purchased this product")
}
2016-06-04 21:48:49 +01:00
2016-07-22 09:48:30 +01:00
case .error(let error):
2016-05-18 18:31:17 +07:00
print("Receipt verification failed: \(error)")
}
}
```
#### Auto-Renewable
```swift
let purchaseResult = SwiftyStoreKit.verifySubscription(
type: .autoRenewable,
productId: "com.musevisions.SwiftyStoreKit.Subscription",
inReceipt: receipt)
```
2016-05-24 11:57:31 +07:00
#### Non-Renewing
```swift
// validDuration: time interval in seconds
let purchaseResult = SwiftyStoreKit.verifySubscription(
type: .nonRenewing(validDuration: 3600 * 24 * 30),
productId: "com.musevisions.SwiftyStoreKit.Subscription",
inReceipt: receipt)
```
2016-05-24 11:57:31 +07:00
2017-05-18 10:47:49 +01:00
**Notes**
* The expiration dates are calculated against the receipt date. This is the date of the last successful call to `verifyReceipt`.
* When purchasing subscriptions in sandbox mode, the expiry dates are set just minutes after the purchase date for testing purposes.
#### Purchasing and verifying a subscription
The `verifySubscription` method can be used together with the `purchaseProduct` method to purchase a subscription and check its expiration date, like so:
```swift
let productId = "your-product-id"
SwiftyStoreKit.purchaseProduct(productId, atomically: true) { result in
if case .success(let purchase) = result {
// Deliver content from server, then:
if purchase.needsFinishTransaction {
SwiftyStoreKit.finishTransaction(purchase.transaction)
}
2017-10-11 17:42:33 +02:00
let appleValidator = AppleReceiptValidator(service: .production, sharedSecret: "your-shared-secret")
SwiftyStoreKit.verifyReceipt(using: appleValidator) { result in
if case .success(let receipt) = result {
let purchaseResult = SwiftyStoreKit.verifySubscription(
type: .autoRenewable,
productId: productId,
inReceipt: receipt)
switch purchaseResult {
case .purchased(let expiryDate, let receiptItems):
print("Product is valid until \(expiryDate)")
case .expired(let expiryDate, let receiptItems):
print("Product is expired since \(expiryDate)")
case .notPurchased:
print("This product has never been purchased")
}
} else {
// receipt verification error
}
}
} else {
// purchase error
}
}
```
## Notes
2015-09-04 12:59:55 +01:00
The framework provides a simple block based API with robust error handling on top of the existing StoreKit framework. It does **NOT** persist in app purchases data locally. It is up to clients to do this with a storage solution of choice (i.e. NSUserDefaults, CoreData, Keychain).
2015-09-04 12:07:52 +01:00
## Installation
### CocoaPods
SwiftyStoreKit can be installed as a [CocoaPod](https://cocoapods.org/) and builds as a Swift framework. To install, include this in your Podfile.
2016-03-30 22:47:01 +02:00
```ruby
use_frameworks!
pod 'SwiftyStoreKit'
```
Once installed, just ```import SwiftyStoreKit``` in your classes and you're good to go.
### Carthage
To integrate SwiftyStoreKit into your Xcode project using [Carthage](https://github.com/Carthage/Carthage), specify it in your Cartfile:
2016-03-30 22:47:01 +02:00
```ogdl
github "bizz84/SwiftyStoreKit"
```
**NOTE**: Please ensure that you have the [latest](https://github.com/Carthage/Carthage/releases) Carthage installed.
2017-06-08 08:57:47 +01:00
## Swift 2.x / 3.x / 4.x
2016-10-20 21:14:31 +01:00
| Language | Branch | Pod version | Xcode version |
| --------- | ------ | ----------- | ------------- |
2017-10-11 20:07:09 +02:00
| Swift 4.x | [master](https://github.com/bizz84/SwiftyStoreKit/tree/master) | >= 0.10.4 | Xcode 9 or greater|
2017-06-08 08:57:47 +01:00
| Swift 3.x | [master](https://github.com/bizz84/SwiftyStoreKit/tree/master) | >= 0.5.x | Xcode 8.x |
2016-10-20 21:14:31 +01:00
| Swift 2.3 | [swift-2.3](https://github.com/bizz84/SwiftyStoreKit/tree/swift-2.3) | 0.4.x | Xcode 8, Xcode 7.3.x |
| Swift 2.2 | [swift-2.2](https://github.com/bizz84/SwiftyStoreKit/tree/swift-2.2) | 0.3.x | Xcode 7.3.x |
## Change Log
See the [Releases Page](https://github.com/bizz84/SwiftyStoreKit/releases)
## Sample Code
The project includes demo apps [for iOS](https://github.com/bizz84/SwiftyStoreKit/blob/master/SwiftyStoreKit-iOS-Demo/ViewController.swift) [and macOS](https://github.com/bizz84/SwiftyStoreKit/blob/master/SwiftyStoreKit-macOS-Demo/ViewController.swift) showing how to use SwiftyStoreKit.
Note that the pre-registered in app purchases in the demo apps are for illustration purposes only and may not work as iTunes Connect may invalidate them.
2015-09-04 12:07:52 +01:00
#### Features
2017-01-21 18:18:16 +00:00
2015-09-04 12:09:35 +01:00
- Super easy to use block based API
- Support for consumable, non-consumable in-app purchases
- Support for free, auto renewable and non renewing subscriptions
2015-09-04 12:04:10 +01:00
- Receipt verification
2016-11-05 00:00:12 +00:00
- iOS, tvOS and macOS compatible
2017-12-28 21:54:51 +00:00
## Video Tutorials
[@rebeloper](https://github.com/rebeloper) has created a step-by-step video guide to IAP / SwiftyStoreKit on YouTube:
#### [Ultimate In-app Purchases Guide | Xcode 9, Swift 4, iOS](https://www.youtube.com/watch?v=bIyj6BZ1-Qw&list=PL_csAAO9PQ8b9kqrltk2_SpYslTwyrwjb)
## Essential Reading
* [Apple - WWDC16, Session 702: Using Store Kit for In-app Purchases with Swift 3](https://developer.apple.com/videos/play/wwdc2016/702/)
* [Apple - TN2387: In-App Purchase Best Practices](https://developer.apple.com/library/content/technotes/tn2387/_index.html)
2017-07-16 14:55:13 +01:00
* [Apple - TN2413: In-App Purchase FAQ](https://developer.apple.com/library/content/technotes/tn2413/_index.html)
* [Apple - About Receipt Validation](https://developer.apple.com/library/content/releasenotes/General/ValidateAppStoreReceipt/Introduction.html)
* [Apple - Receipt Validation Programming Guide](https://developer.apple.com/library/content/releasenotes/General/ValidateAppStoreReceipt/Chapters/ReceiptFields.html#//apple_ref/doc/uid/TP40010573-CH106-SW1)
* [Apple - Validating Receipts Locally](https://developer.apple.com/library/content/releasenotes/General/ValidateAppStoreReceipt/Chapters/ValidateLocally.html)
* [Apple - Working with Subscriptions](https://developer.apple.com/library/content/documentation/NetworkingInternet/Conceptual/StoreKitGuide/Chapters/Subscriptions.html#//apple_ref/doc/uid/TP40008267-CH7-SW6)
* [Apple - Offering Subscriptions](https://developer.apple.com/app-store/subscriptions/)
* [Apple - Restoring Purchased Products](https://developer.apple.com/library/content/documentation/NetworkingInternet/Conceptual/StoreKitGuide/Chapters/Restoring.html#//apple_ref/doc/uid/TP40008267-CH8-SW9)
2017-08-22 17:26:33 +01:00
* [Apple - Testing In-App Purchase Products](https://developer.apple.com/library/content/documentation/LanguagesUtilities/Conceptual/iTunesConnectInAppPurchase_Guide/Chapters/TestingInAppPurchases.html): includes info on duration of subscriptions in sandbox mode
* [objc.io - Receipt Validation](https://www.objc.io/issues/17-security/receipt-validation/)
I have also written about building SwiftyStoreKit on Medium:
* [How I got 1000 ⭐️ on my GitHub Project](https://medium.com/ios-os-x-development/how-i-got-1000-%EF%B8%8F-on-my-github-project-654d3d394ca6#.1idp27olf)
* [Maintaining a Growing Open Source Project](https://medium.com/@biz84/maintaining-a-growing-open-source-project-1d385ca84c5#.4cv2g7tdc)
2017-10-11 16:10:55 +02:00
### Troubleshooting
* [Apple TN 2413 - Why are my product identifiers being returned in the invalidProductIdentifiers array?](https://developer.apple.com/library/content/technotes/tn2413/_index.html#//apple_ref/doc/uid/DTS40016228-CH1-TROUBLESHOOTING-WHY_ARE_MY_PRODUCT_IDENTIFIERS_BEING_RETURNED_IN_THE_INVALIDPRODUCTIDENTIFIERS_ARRAY_)
* [Invalid Product IDs](http://troybrant.net/blog/2010/01/invalid-product-ids/): Checklist of common mistakes
* [Testing Auto-Renewable Subscriptions on iOS](http://davidbarnard.com/post/164337147440/testing-auto-renewable-subscriptions-on-ios)
* [Apple forums - iOS 11 beta sandbox - cannot connect to App Store](https://forums.developer.apple.com/message/261428#261428)
2017-01-21 18:18:16 +00:00
## Payment flows - implementation Details
2015-09-04 11:05:00 +01:00
In order to make a purchase, two operations are needed:
2017-01-21 18:18:16 +00:00
- Perform a `SKProductRequest` to obtain the `SKProduct` corresponding to the product identifier.
2016-06-04 21:48:49 +01:00
2017-01-21 18:18:16 +00:00
- Submit the payment and listen for updated transactions on the `SKPaymentQueue`.
2015-09-04 11:49:43 +01:00
The framework takes care of caching SKProducts so that future requests for the same `SKProduct` don't need to perform a new `SKProductRequest`.
2015-09-04 11:05:00 +01:00
2017-01-21 18:18:16 +00:00
### Payment queue
2017-01-21 18:18:16 +00:00
The following list outlines how requests are processed by SwiftyStoreKit.
2015-09-04 11:05:00 +01:00
2017-01-21 18:18:16 +00:00
* `SKPaymentQueue` is used to queue payments or restore purchases requests.
* Payments are processed serially and in-order and require user interaction.
* Restore purchases requests don't require user interaction and can jump ahead of the queue.
* `SKPaymentQueue` rejects multiple restore purchases calls.
2017-04-24 14:33:01 +01:00
* Failed transactions only ever belong to queued payment requests.
2017-01-21 18:18:16 +00:00
* `restoreCompletedTransactionsFailedWithError` is always called when a restore purchases request fails.
* `paymentQueueRestoreCompletedTransactionsFinished` is always called following 0 or more update transactions when a restore purchases request succeeds.
* A complete transactions handler is require to catch any transactions that are updated when the app is not running.
* Registering a complete transactions handler when the app launches ensures that any pending transactions can be cleared.
* If a complete transactions handler is missing, pending transactions can be mis-attributed to any new incoming payments or restore purchases.
2015-09-04 11:05:00 +01:00
2017-01-21 18:18:16 +00:00
The order in which transaction updates are processed is:
2015-09-04 11:05:00 +01:00
2017-01-21 18:18:16 +00:00
1. payments (transactionState: `.purchased` and `.failed` for matching product identifiers)
2. restore purchases (transactionState: `.restored`, or `restoreCompletedTransactionsFailedWithError`, or `paymentQueueRestoreCompletedTransactionsFinished`)
3. complete transactions (transactionState: `.purchased`, `.failed`, `.restored`, `.deferred`)
2015-09-04 11:05:00 +01:00
Any transactions where state is `.purchasing` are ignored.
2015-09-04 11:05:00 +01:00
See [this pull request](https://github.com/bizz84/SwiftyStoreKit/pull/131) for full details about how the payment flows have been implemented.
## Credits
2016-11-05 00:00:12 +00:00
Many thanks to [phimage](https://github.com/phimage) for adding macOS support and receipt verification.
## Apps using SwiftyStoreKit
It would be great to showcase apps using SwiftyStoreKit here. Pull requests welcome :)
2016-04-24 18:18:56 +03:00
* [MDacne](https://itunes.apple.com/app/id1044050208) - Acne analysis and treatment
* [Pixel Picker](https://itunes.apple.com/app/id930804327) - Image Color Picker
* [KType](https://itunes.apple.com/app/id1037000234) - Space shooter game
* [iPic](https://itunes.apple.com/app/id1101244278) - Automatically upload images and save Markdown links
* [iHosts](https://itunes.apple.com/app/id1102004240) - Perfect for editing /etc/hosts
* [Arise](http://www.abnehm-app.de/) - Calorie counter
* [Truth Truth Lie](https://itunes.apple.com/app/id1130832864) - iMessage game, featured by Apple
* [Tactus Music Player](https://itunes.apple.com/app/id557446352) - Alternative music player app
* [Drops](https://itunes.apple.com/app/id939540371) - Language learning app
2017-02-17 18:45:09 -07:00
* [Fresh Snow](https://itunes.apple.com/app/id1063000470) - Colorado Ski Report
* [Zmeu Grand Canyon](http://grandcanyon.zmeu.guide/) - Interactive hiking map & planner
2017-07-06 10:14:46 +01:00
* [OB Monitor](https://itunes.apple.com/app/id1073398446) - The app for Texas Longhorns athletics fans
2017-11-11 20:40:53 -08:00
* [Talk Dim Sum](https://itunes.apple.com/us/app/talk-dim-sum/id953929066) - Your dim sum companion
2015-09-04 12:09:35 +01:00
## License
2015-09-04 11:05:00 +01:00
2017-01-21 18:22:14 +00:00
Copyright (c) 2015-2017 Andrea Bizzotto bizz84@gmail.com
2015-09-04 12:09:35 +01:00
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.