-
Notifications
You must be signed in to change notification settings - Fork 116
/
Copy pathUnderlyingError+Stripe.swift
251 lines (249 loc) · 11 KB
/
UnderlyingError+Stripe.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
#if !targetEnvironment(macCatalyst)
import StripeTerminal
/// Maps Stripe SDK specific errors to domain errors:
/// the mapping is done according to the error codes documented here:
/// https://stripe.dev/stripe-terminal-ios/docs/Enums/SCPError.html
extension UnderlyingError {
init?(withStripeError stripeError: Error) {
let error = stripeError as NSError
guard error.domain == ErrorDomain else {
return nil
}
guard let stripeError = StripeTerminal.ErrorCode.Code(rawValue: error.code) else {
return nil
}
switch stripeError {
case .notConnectedToReader:
self = .notConnectedToReader
case .alreadyConnectedToReader:
self = .alreadyConnectedToReader
case .confirmInvalidPaymentIntent:
self = .confirmInvalidPaymentIntent
case .unsupportedSDK:
self = .unsupportedSDK
case .featureNotAvailableWithConnectedReader:
self = .featureNotAvailableWithConnectedReader
case .canceled:
self = .commandCancelled(from: .unknown)
case .locationServicesDisabled:
self = .locationServicesDisabled
case .bluetoothDisabled:
self = .bluetoothDisabled
case .bluetoothError:
self = .bluetoothError
case .bluetoothScanTimedOut:
self = .bluetoothScanTimedOut
case .bluetoothLowEnergyUnsupported:
self = .bluetoothLowEnergyUnsupported
case .bluetoothConnectionFailedBatteryCriticallyLow:
self = .bluetoothConnectionFailedBatteryCriticallyLow
case .readerSoftwareUpdateFailedBatteryLow:
self = .readerSoftwareUpdateFailedBatteryLow
case .readerSoftwareUpdateFailedInterrupted:
self = .readerSoftwareUpdateFailedInterrupted
case .readerSoftwareUpdateFailed:
self = .readerSoftwareUpdateFailed
case .readerSoftwareUpdateFailedReaderError:
self = .readerSoftwareUpdateFailedReader
case .readerSoftwareUpdateFailedServerError:
self = .readerSoftwareUpdateFailedServer
case .cardInsertNotRead:
self = .cardInsertNotRead
case .cardSwipeNotRead:
self = .cardSwipeNotRead
case .cardReadTimedOut:
self = .cardReadTimeOut
case .cardRemoved:
self = .cardRemoved
case .cardLeftInReader:
self = .cardLeftInReader
case .readerBusy:
self = .readerBusy
case .incompatibleReader:
self = .readerIncompatible
case .readerCommunicationError:
self = .readerCommunicationError
case .bluetoothConnectTimedOut:
self = .bluetoothConnectTimedOut
case .bluetoothDisconnected:
self = .bluetoothDisconnected
case .unsupportedReaderVersion:
self = .unsupportedReaderVersion
case .connectFailedReaderIsInUse:
self = .connectFailedReaderIsInUse
case .unexpectedSdkError:
self = .unexpectedSDKError
case .declinedByStripeAPI:
// https://stripe.dev/stripe-terminal-ios/docs/Errors.html#/c:@SCPErrorKeyStripeAPIDeclineCode
let declineCode = error.userInfo[ErrorKey.stripeAPIDeclineCode.rawValue] as? String
let declineReason = DeclineReason(with: declineCode ?? "")
self = .paymentDeclinedByPaymentProcessorAPI(declineReason: declineReason)
case .declinedByReader:
self = .paymentDeclinedByCardReader
case .notConnectedToInternet:
self = .notConnectedToInternet
case .requestTimedOut:
self = .requestTimedOut
case .sessionExpired:
self = .readerSessionExpired
case .stripeAPIError:
self = .processorAPIError
case .passcodeNotEnabled:
self = .passcodeNotEnabled
case .tapToPayReaderTOSAcceptanceRequiresiCloudSignIn:
self = .tapToPayReaderTOSAcceptanceRequiresiCloudSignIn
case .nfcDisabled:
self = .nfcDisabled
case .tapToPayReaderFailedToPrepare:
self = .tapToPayReaderFailedToPrepare
case .tapToPayReaderTOSAcceptanceCanceled:
self = .tapToPayReaderTOSAcceptanceCanceled
case .tapToPayReaderTOSNotYetAccepted:
self = .tapToPayReaderTOSNotYetAccepted
case .tapToPayReaderTOSAcceptanceFailed:
self = .tapToPayReaderTOSAcceptanceFailed
case .tapToPayReaderMerchantBlocked:
self = .tapToPayReaderMerchantBlocked
case .tapToPayReaderInvalidMerchant:
self = .tapToPayReaderInvalidMerchant
case .tapToPayReaderDeviceBanned:
self = .tapToPayReaderDeviceBanned
case .unsupportedMobileDeviceConfiguration:
self = .unsupportedMobileDeviceConfiguration
case .readerNotAccessibleInBackground:
self = .readerNotAccessibleInBackground
case .commandNotAllowedDuringCall:
self = .commandNotAllowedDuringCall
case .invalidAmount:
self = .invalidAmount
case .invalidCurrency:
self = .invalidCurrency
case .cancelFailedAlreadyCompleted:
self = .cancelFailedAlreadyCompleted
case .nilPaymentIntent:
self = .nilPaymentIntent
case .nilSetupIntent:
self = .nilSetupIntent
case .nilRefundPaymentMethod:
self = .nilRefundPaymentMethod
case .invalidRefundParameters:
self = .invalidRefundParameters
case .invalidClientSecret:
self = .invalidClientSecret
case .invalidDiscoveryConfiguration:
self = .invalidDiscoveryConfiguration
case .invalidReaderForUpdate:
self = .invalidReaderForUpdate
case .featureNotAvailable:
self = .featureNotAvailable
case .bluetoothConnectionInvalidLocationIdParameter:
self = .bluetoothConnectionInvalidLocationIdParameter
case .invalidRequiredParameter:
self = .invalidRequiredParameter
case .forwardingTestModePaymentInLiveMode:
self = .forwardingTestModePaymentInLiveMode
case .forwardingLiveModePaymentInTestMode:
self = .forwardingLiveModePaymentInTestMode
case .readerConnectionConfigurationInvalid:
self = .readerConnectionConfigurationInvalid
case .readerTippingParameterInvalid:
self = .readerTippingParameterInvalid
case .invalidLocationIdParameter:
self = .invalidLocationIdParameter
case .bluetoothAccessDenied:
self = .bluetoothDenied
case .readerSoftwareUpdateFailedExpiredUpdate:
self = .readerSoftwareUpdateFailedExpiredUpdate
case .missingEMVData:
self = .missingEMVData
case .commandNotAllowed:
self = .commandNotAllowed
case .bluetoothPeerRemovedPairingInformation:
self = .bluetoothPeerRemovedPairingInformation
case .bluetoothAlreadyPairedWithAnotherDevice:
self = .bluetoothAlreadyPairedWithAnotherDevice
case .unknownReaderIpAddress:
self = .unknownReaderIpAddress
case .internetConnectTimeOut:
self = .internetConnectTimeOut
case .bluetoothReconnectStarted:
self = .bluetoothReconnectStarted
case .tapToPayReaderAccountDeactivated:
self = .tapToPayReaderAccountDeactivated
case .readerMissingEncryptionKeys:
self = .readerMissingEncryptionKeys
case .unexpectedReaderError:
self = .unexpectedReaderError
case .commandRequiresCardholderConsent:
self = .commandRequiresCardholderConsent
case .refundFailed:
self = .refundFailed
case .cardSwipeNotAvailable:
self = .cardSwipeNotAvailable
case .interacNotSupportedOffline:
self = .interacNotSupportedOffline
case .offlineAndCardExpired:
self = .offlineAndCardExpired
case .offlineTransactionDeclined:
self = .offlineTransactionDeclined
case .offlineCollectAndConfirmMismatch:
self = .offlineCollectAndConfirmMismatch
case .onlinePinNotSupportedOffline:
self = .onlinePinNotSupportedOffline
case .offlineTestCardInLivemode:
self = .offlineTestCardInLivemode
case .stripeAPIResponseDecodingError:
self = .stripeAPIResponseDecodingError
case .internalNetworkError:
self = .internalNetworkError
case .connectionTokenProviderCompletedWithError:
self = .connectionTokenProviderCompletedWithError
case .connectionTokenProviderTimedOut:
self = .connectionTokenProviderTimedOut
/// Our `DefaultConnectionTokenProvider` implementation of `fetchConnectionToken` never calls completion block with `(nil, nil)`.
case .connectionTokenProviderCompletedWithNothing,
/// Offline mode is not supported.
.connectionTokenProviderCompletedWithNothingWhileForwarding,
.accountIdMismatchWhileForwarding,
.updatePaymentIntentUnavailableWhileOffline,
.updatePaymentIntentUnavailableWhileOfflineModeEnabled,
.offlinePaymentsDatabaseTooLarge,
.readerConnectionOfflinePairingUnseenDisabled,
.noLastSeenAccount,
.connectionTokenProviderCompletedWithErrorWhileForwarding,
.offlineBehaviorForceOfflineWithFeatureDisabled,
.readerConnectionNotAvailableOffline,
.readerConnectionOfflineLocationMismatch,
.readerConnectionOfflineNeedsUpdate,
.amountExceedsMaxOfflineAmount,
.invalidOfflineCurrency,
.encryptionKeyFailure,
.encryptionKeyStillInitializing,
.notConnectedToInternetAndOfflineBehaviorRequireOnline,
/// We don’t request a list of locations directly, but request the store location instead.
.invalidListLocationsLimitParameter,
/// `on_behalf_of` parameter is not set in the payment intent.
.invalidRequiredParameterOnBehalfOf,
/// Dynamic currency conversion not supported.
.requestDynamicCurrencyConversionRequiresUpdatePaymentIntent,
.dynamicCurrencyConversionNotAvailable,
/// Surcharging is not supported.
.surchargingNotAvailable,
.surchargeNoticeRequiresUpdatePaymentIntent,
.surchargeUnavailableWithDynamicCurrencyConversion,
/// Collecting on-screen inputs from card reader is not supported.
.collectInputsInvalidParameter,
.collectInputsUnsupported,
.collectInputsTimedOut,
.collectInputsApplicationError,
/// USB discovery is not supported.
.usbDiscoveryTimedOut,
.usbDisconnected:
assertionFailure("Unexpected Stripe error that we should consider handling: \(stripeError)")
return nil
default:
return nil
}
}
}
#endif