React Native / Swift-应用内购买代码在沙箱和testflight中随机崩溃

布里恩·克里恩(Brien Crean)

我正在尝试将本机应用程序包含在应用程序购买中。我已经尝试过react-native-in-app-utils并在Swift中创建一个本机模块。两者都可以工作,但看起来都非常不稳定。Xcode控制台中的随机崩溃没有错误。试飞时稳定得多,但仍然会崩溃,并且似乎没有任何押韵或理由。崩溃的大部分发生在购买上,但在从Apple取回购买或恢复购买时也发生了很多。这是我的快速代码,我可以使用Objective C桥来响应本机代码:

// Create Product List
struct ProductList {
  static let monthlySub       : String  = "monthlysubscription"
  static let threeMonthlySub  : String  = "threemonthlysubscription"
  static let sixMonthlySub    : String  = "sixmonthlysubscription"
  static let yearlySub        : String  = "yearlysubscription"

  static let products = [monthlySub, threeMonthlySub, sixMonthlySub, yearlySub]
}

@objc(StoreManager)
class StoreManager: NSObject {

  var loadedProducts: Dictionary<String, SKProduct> = [:]
  // callback holders defined as optionals
  var purchaseCallbackHolder: RCTResponseSenderBlock? = nil
  var productsCallbackHolder: RCTResponseSenderBlock? = nil
  var restoreCallbackHolder: RCTResponseSenderBlock? = nil

  override init() {
    super.init()
    SKPaymentQueue.defaultQueue().addTransactionObserver(self)
  }

  @objc func getProductList(name: String, callback successCallback: RCTResponseSenderBlock){
    // check payments allowed
    if SKPaymentQueue.canMakePayments() {
      // products from static struct
      let products = NSSet(array: ProductList.products);
      // When request completes, calls the SKProductsRequestDelegate
      let request = SKProductsRequest(productIdentifiers: products as! Set<String>);
      productsCallbackHolder = successCallback
      request.delegate = self;
      request.start();
    }
  }

  @objc func purchaseProduct(productIdentifier: String, callback successCallback: RCTResponseSenderBlock) {
    let product = loadedProducts[productIdentifier as String]!
    let payment = SKPayment(product: product)
    // add callback to holder
    purchaseCallbackHolder = successCallback
    // Triggers SKPaymentTransactionObserver
    SKPaymentQueue.defaultQueue().addPayment(payment)
  }

  @objc func restorePurchases(name:String, callback successCallback: RCTResponseSenderBlock){
    restoreCallbackHolder = successCallback
    print("Restoring Purchases")
    SKPaymentQueue.defaultQueue().restoreCompletedTransactions()
  }


  @objc func validatePurchases(name:String, callback successCallback: RCTResponseSenderBlock) -> Void {

    let receiptUrl = NSBundle.mainBundle().appStoreReceiptURL
    let receipt: NSData = NSData(contentsOfURL:receiptUrl!)!
    let receiptdata: NSString = receipt.base64EncodedStringWithOptions(NSDataBase64EncodingOptions(rawValue: 0))

    // Pass Base64 encoded string back to JS
    successCallback([receiptdata])

  }

  func updateWithProducts(products: [SKProduct]) {

    var productIdentifiers: Dictionary<String, NSNumber> = [:]
    for product in products {
      loadedProducts[product.productIdentifier] = product
      productIdentifiers[product.productIdentifier] = product.price
    }
    productsCallbackHolder?([productIdentifiers])
  }


}

extension StoreManager: SKProductsRequestDelegate {
  func productsRequest(request: SKProductsRequest, didReceiveResponse response: SKProductsResponse) {
    // products retrieved from App store
    let appProducts = response.products
    if appProducts.count != 0 {
      for product in appProducts{
        print(product.productIdentifier)
      }
      print(appProducts)
      updateWithProducts(appProducts)
    }
    else {
      // return error/info to react native
      print("no products received from store")
    }
  }
}
extension StoreManager: SKPaymentTransactionObserver {
  func paymentQueue(queue: SKPaymentQueue, updatedTransactions transactions: [SKPaymentTransaction])    {
    print("Received Payment Transaction Response from Apple");
    for transaction:AnyObject in transactions {
      // check object is a transaction first
      if let trans:SKPaymentTransaction = transaction as? SKPaymentTransaction{
        switch trans.transactionState {
        case .Purchased:
          print("Product Purchased");
          SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
          break;
        case .Failed:
          print("Purchased Failed");
          SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
          break;
        case .Restored:
          print("Purchases Restored");
          SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
        default:
          break;
        }
      }
      // invoke any callback waiting to be called
      purchaseCallbackHolder?([])
      purchaseCallbackHolder = nil
      restoreCallbackHolder?([])
      restoreCallbackHolder = nil
    }

  }
}

任何帮助将非常感激!

更新:我有下面的崩溃的堆栈跟踪

AppName[1788:581487] *** -[StoreManager respondsToSelector:]: message sent to deallocated instance 0x14d43f80
(lldb) bt
* thread #4: tid = 0x8df6f, 0x25750ffe CoreFoundation`___forwarding___ + 530, queue = 'com.facebook.react.StoreManagerQueue', stop reason = EXC_BREAKPOINT (code=EXC_ARM_BREAKPOINT, subcode=0xdefe)
  * frame #0: 0x25750ffe CoreFoundation`___forwarding___ + 530
    frame #1: 0x2567b298 CoreFoundation`_CF_forwarding_prep_0 + 24
    frame #2: 0x2f103f1e StoreKit`__NotifyObserverAboutChanges + 66
    frame #3: 0x2565bc08 CoreFoundation`CFArrayApplyFunction + 36
    frame #4: 0x2f103ecc StoreKit`-[SKPaymentQueue _notifyObserversAboutChanges:sendUpdatedDownloads:] + 128
    frame #5: 0x2f1028e8 StoreKit`-[SKPaymentQueue addPayment:] + 320
    frame #6: 0x00106438 AppName`StoreManager.purchaseProduct(productIdentifier="iosmonthlysubscription799", successCallback=0x00106694 AppName`partial apply forwarder for reabstraction thunk helper from @callee_unowned @convention(block) (@unowned Swift.ImplicitlyUnwrappedOptional<__ObjC.NSArray>) -> (@unowned ()) to @callee_owned (@owned Swift.ImplicitlyUnwrappedOptional<Swift.Array<Swift.AnyObject>>) -> (@unowned ()) with unmangled suffix ".40" at inAppPayments.swift, self=0x14d0a990) -> ()) -> () + 1168 at inAppPayments.swift:54
    frame #7: 0x00106634 AppName`@objc StoreManager.purchaseProduct(String, callback : ([AnyObject]!) -> ()) -> () + 240 at inAppPayments.swift:0
    frame #8: 0x25752664 CoreFoundation`__invoking___ + 68
    frame #9: 0x256778bc CoreFoundation`-[NSInvocation invoke] + 292
    frame #10: 0x2567b356 CoreFoundation`-[NSInvocation invokeWithTarget:] + 50
    frame #11: 0x001672d6 AppName`-[RCTModuleMethod invokeWithBridge:module:arguments:](self=0x14fdba90, _cmd="invokeWithBridge:module:arguments:", bridge=0x14faa1a0, module=0x14d0a990, arguments=@"2 elements") + 1636 at RCTModuleMethod.m:489
    frame #12: 0x001abaea AppName`-[RCTBatchedBridge _handleRequestNumber:moduleID:methodID:params:](self=0x14faa1a0, _cmd="_handleRequestNumber:moduleID:methodID:params:", i=6, moduleID=77, methodID=3, params=@"2 elements") + 926 at RCTBatchedBridge.m:987
    frame #13: 0x001aae44 AppName`__33-[RCTBatchedBridge handleBuffer:]_block_invoke.452(.block_descriptor=<unavailable>) + 1028 at RCTBatchedBridge.m:915
    frame #14: 0x00b9bba6 libdispatch.dylib`_dispatch_call_block_and_release + 10
    frame #15: 0x00ba64aa libdispatch.dylib`_dispatch_queue_drain + 2014
    frame #16: 0x00b9ede2 libdispatch.dylib`_dispatch_queue_invoke + 282
    frame #17: 0x00ba799e libdispatch.dylib`_dispatch_root_queue_drain + 426
    frame #18: 0x00ba77f0 libdispatch.dylib`_dispatch_worker_thread3 + 100
    frame #19: 0x2547ee0c libsystem_pthread.dylib`_pthread_wqthread + 1024
    frame #20: 0x2547e9fc libsystem_pthread.dylib`start_wqthread + 8

好像是因为我的回调引用是可选的而崩溃了?这是由Objective C桥调用的Swift。我相信目标C没有可选概念。

var purchaseCallbackHolder: RCTResponseSenderBlock? = nil

有一个更好的方法吗?再次感谢

布里恩·克里恩(Brien Crean)

事务观察者需要通过deinit方法删除:

  override init() {
    super.init()
    SKPaymentQueue.defaultQueue().addTransactionObserver(self)
  }


  deinit {
    if SKPaymentQueue.canMakePayments() {
      SKPaymentQueue.defaultQueue().removeTransactionObserver(self)
    }
  }

我查找了我在堆栈跟踪中遇到的错误,其中一个是:

StoreKit__NotifyObserverAboutChanges

向我指出了这篇文章:StoreKit代码中的一个错误使我失去了90%的IAP销售

本文收集自互联网,转载请注明来源。

如有侵权,请联系[email protected] 删除。

编辑于
0

我来说两句

0条评论
登录后参与评论

相关文章

来自分类Dev

Swift 中的 React-native iOS 桥接模块

来自分类Dev

将 swift 块传递给 React Native

来自分类Dev

React Native Swift UI 组件框架大小

来自分类Dev

流星和React Native

来自分类Dev

React Native中的zIndex

来自分类Dev

React Native中的海拔

来自分类Dev

在React Native中链接

来自分类Dev

第一个应用程序:Swift,Obj-C或React Native?

来自分类Dev

如何在React Native或Swift中实现IG TV底部滑动面板?

来自分类Dev

无法在Swift中创建React Native项目。链接器不会链接到当前架构

来自分类Dev

如何获取React Native应用的崩溃日志?

来自分类Dev

android native和react native之间的通信

来自分类Dev

React Native App 突然崩溃

来自分类Dev

在发布中调试 React Native 崩溃

来自分类Dev

React Native-构建应用

来自分类Dev

React Native Promises令人困惑的Swift类型注释

来自分类Dev

如何将Swift Promise桥接到React Native

来自分类Dev

Xcode 10.2 iOS Swift(+ React Native)项目编译错误

来自分类Dev

React Native和{flex:1}

来自分类Dev

React Native和require('http')

来自分类Dev

React Native和Firebase 3

来自分类Dev

React Native - android 和 iOS?

来自分类Dev

Promises 和 React Native 协助

来自分类Dev

Flex 和布局 (React Native)

来自分类Dev

结合React和React-Native

来自分类Dev

在 React 和 React Native 中使用模型

来自分类Dev

如何在React Native中安装React Native Maps?

来自分类Dev

在React Native中渲染HTML

来自分类Dev

React Native中的透明覆盖