Crash on SKQueueManager.receivedFailedTransaction
Vaberer opened this issue · 7 comments
Operating System: iOS 17.2.1
Device: Phone 15 Pro Max
SDK Version: Adapty 2.7.0
Crashed: Fatal Exception: NSRangeException
[__NSArrayM removeObjectsInRange:]: range {0, 1} extends beyond bounds for empty array
Fatal Exception: NSRangeException
0 CoreFoundation 0xec69c __exceptionPreprocess
1 libobjc.A.dylib 0x2bc80 objc_exception_throw
2 CoreFoundation 0x39228 -[__NSArrayM removeObjectsInRange:]
3 StoreKit 0x33808 -[SKPaymentQueue _removeLocalTransaction:]
4 StoreKit 0x2db58 -[SKPaymentQueue finishTransaction:]
5 Adapty 0xea028 closure #1 in SKQueueManager.receivedFailedTransaction(_:) + 40 (SKQueueManager+MakePurchase.swift:40)
6 Adapty 0x12ab0 thunk for @escaping @callee_guaranteed @Sendable () -> () (<compiler-generated>)
7 libdispatch.dylib 0x26a8 _dispatch_call_block_and_release
8 libdispatch.dylib 0x4300 _dispatch_client_callout
9 libdispatch.dylib 0xb894 _dispatch_lane_serial_drain
10 libdispatch.dylib 0xc3c4 _dispatch_lane_invoke
11 libdispatch.dylib 0x17004 _dispatch_root_queue_drain_deferred_wlh
12 libdispatch.dylib 0x16878 _dispatch_workloop_worker_thread
13 libsystem_pthread.dylib 0x1964 _pthread_wqthread
14 libsystem_pthread.dylib 0x1a04 start_wqthread
This crash occurred for App Store users.
Hi, @Vaberer! Thank you for the issue. According to the stack trace you have provided, it looks like this crash was occurred within StoreKit framework. How often do you observe this behaviour? Are there any other details you can provide?
Hello @x401om thanks for your reply, maybe I found the root cause, I'm also observing SKPaymentTransactionObserver
in the codebase and Adapty does the same and we both are calling SKPaymentQueue.default().finishTransaction(transaction)
.
As I understand, I should pass true
value for observerMode
https://docs.adapty.io/docs/ios-observer-mode
Am. I correct? if so, I'd like to confirm my current implementation, when a user wants to buy something, I'm calling
Adapty.makePurchase(product: product) { _ in
// No need to handle because app has it's own observer.
}
and finishing transactions are made in the app layer. We need to integrate Adapty due to analytics and pulling the products from Adapty paywalls. Is my understanding correct?
Thanks
@Vaberer, you are correct. If you don't want the Adapty SDK to finish transactions, you have to activate it in observer mode. It seems like StoreKit is not very reliable at finishing the same transaction twice, as we can see.
By the way, I am curious, what is the reason you want to finish transactions by yourself? Observer Mode is intended to be used as you described, but we did not expect users to use our products and the makePurchase method in that mode, since they are supposed to have their own purchase infrastructure. There is no issue if you use makePurchase, but I have to ask for the reasons, and perhaps you will consider using the Adapty SDK in full mode.
@x401om let me explain our situation and maybe you can help.
We cooperate with a marketing agency and they want us to implement Adapty for analytics, pulling products and running ab tests.
However we, originally, had our own system for managing in-app purchases and we had to make our and Adapty systems work together. Our systems also have our own analytics and logic (i.e: sending receipts to our backend) to make everything work.
One idea came to my mind:
We still add an observer SKPaymentQueue.default().add(self)
but we won't call SKPaymentQueue.default().finishTransaction(transaction)
.
We just need to listen to the delegate methods to send data to our backend and do our own analytics.
Is this approach correct? Will Adapty analytics and ab testing work?
Thanks.
@Vaberer In Observer Mode, the only difference lies in calling the finishTransaction
method. You can fully rely on our infrastructure for purchases, and, of course, you can listen to the SKPaymentQueue
yourself, but refrain from closing transactions.
Additionally, you can utilize the getReceipt
method and the transaction
property of the AdaptyPurchasedInfo
object, which is the result of makePurchase
.
in this case we can close the issue, thank you for your help
Operating System: iOS 17.4.1
Device: Phone 12 Pro Max
SDK Version: Adapty 2.10.3
Fatal Exception: NSRangeException
*** -[__NSArrayM removeObjectsInRange:]: range {0, 1} extends beyond bounds for empty array
SK1QueueManager+MakePurchase.swift - Line 40 closure #1 in SK1QueueManager.receivedFailedTransaction(_:) + 40
This crash affected users of App Store. But we don't observe and don't finish the transactions, in contrast to the previous justification. All we do is use Adapty throughout the entire purchasing process.