如何在生产中获得原始应用程序版本?

How can you get Original Application Version in Production?

我们最近将一个可购买的应用程序转换为 "freemium" 模型。我们正在使用 Bundle.main.appStoreReceiptURL 来提取收据,然后检查 "original_application_version" 以查看用户是否从 App Store 下载了较旧的付费版本,或者他们是否下载了带有非消耗品的较新的免费版本购买应用升级至完整版。

这在测试 Sandbox 时非常有效,但在生产中,旧版本的应用程序无法正确验证它们是在免费增值版本之前下载的。

使用 productionStoreURL 调用以下代码并从 Bundle.main.appStoreReceiptURL 获得收据:

private let productionStoreURL = URL(string: "https://buy.itunes.apple.com/verifyReceipt")
private let sandboxStoreURL = URL(string: "https://sandbox.itunes.apple.com/verifyReceipt")

private func verifyIfPurchasedBeforeFreemium(_ storeURL: URL, _ receipt: Data) {
    do {
        let requestContents:Dictionary = ["receipt-data": receipt.base64EncodedString()]
        let requestData = try JSONSerialization.data(withJSONObject: requestContents, options: [])

        var storeRequest = URLRequest(url: storeURL)
        storeRequest.httpMethod = "POST"
        storeRequest.httpBody = requestData

        URLSession.shared.dataTask(with: storeRequest) { (data, response, error) in
            DispatchQueue.main.async {
                if data != nil {
                    do {
                        let jsonResponse = try JSONSerialization.jsonObject(with: data!, options: []) as! [String: Any?]

                        if let statusCode = jsonResponse["status"] as? Int {
                            if statusCode == 21007 {
                                print("Switching to test against sandbox")
                                self.verifyIfPurchasedBeforeFreemium(self.sandboxStoreURL!, receipt)
                            }
                        }

                        if let receiptResponse = jsonResponse["receipt"] as? [String: Any?], let originalVersion = receiptResponse["original_application_version"] as? String {
                            if self.isPaidVersionNumber(originalVersion) {
                                // Update to full paid version of app
                                UserDefaults.standard.set(true, forKey: upgradeKeys.isUpgraded)
                                NotificationCenter.default.post(name: .UpgradedVersionNotification, object: nil)
                            }
                        }
                    } catch {
                        print("Error: " + error.localizedDescription)
                    }
                }
            }
            }.resume()
    } catch {
        print("Error: " + error.localizedDescription)
    }
}

private func isPaidVersionNumber(_ originalVersion: String) -> Bool {
    let pattern:String = "^\d+\.\d+"
    do {
        let regex = try NSRegularExpression(pattern: pattern, options: [])
        let results = regex.matches(in: originalVersion, options: [], range: NSMakeRange(0, originalVersion.count))

        let original = results.map {
            Double(originalVersion[Range([=10=].range, in: originalVersion)!])
        }

        if original.count > 0, original[0]! < firstFreemiumVersion {
            print("App purchased prior to Freemium model")
            return true
        }
    } catch {
        print("Paid Version RegEx Error.")
    }
    return false
}

第一个免费增值版本是 3.2,这是我们当前的版本。所有以前的版本都是 3.1.6 或更早版本。

Production URL 应该不是问题,否则它不会回退 21007 状态代码来为我们触发 Sandbox 验证。但是,解决这个问题特别棘手,因为我们无法针对 Apple 的产品 URL 本身进行测试。

有人知道为什么这在沙盒中有效但在生产中无效吗?

看来拿到收据完全没有问题。

original_application_version 的一些旧值的格式不正确,导致我们无法获取要比较的应用程序版本。